@pwrdrvr/microapps-cdk 0.2.12 → 0.2.13

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
@@ -3100,7 +3100,7 @@
3100
3100
  },
3101
3101
  "name": "@pwrdrvr/microapps-cdk",
3102
3102
  "readme": {
3103
- "markdown": "![Build/Deploy CI](https://github.com/pwrdrvr/microapps-core/actions/workflows/ci.yml/badge.svg) ![Main Build](https://github.com/pwrdrvr/microapps-core/actions/workflows/main-build.yml/badge.svg) ![Release](https://github.com/pwrdrvr/microapps-core/actions/workflows/release.yml/badge.svg)\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` that looks up the version of the application to be run, then renders a transparent `iframe` with a link to that version. 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 pattern `[/{prefix}]/{appname}/{semver}/`. These \"versioned\" URLs are not intended to be advertised to end users as they would cause a user to be stuck on a particular version of the app if the URL was bookmarked. Note that the system does not limit access to particular versions of an application, as of 2022-01-26, but that can be added as a feature.\n\n# Table of Contents <!-- omit in toc -->\n\n- [Overview](#overview)\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- [Why MicroApps](#why-microapps)\n- [Limitations / Future Development](#limitations--future-development)\n- [Related Projects / Components](#related-projects--components)\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 - [deploy.json](#deployjson)\n - [serverless.yaml](#serverlessyaml)\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# 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- `cdk-sso-sync`\n - Using `npm i -g cdk-sso-sync`\n - Sets AWS SSO credentials in a way that CDK can use them\n - Not necessary if not using AWS SSO\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:stackName=microapps-demo-deploy --context @pwrdrvr/microapps:deployReleaseApp=true microapps-basic`\n - `npx microapps-publish publish -a release -n ${RELEASE_APP_PACKAGE_VERSION} -d ${DEPLOYER_LAMBDA_NAME} -l ${RELEASE_APP_LAMBDA_NAME} -s node_modules/@pwrdrvr/microapps-app-release-cdk/lib/.static_files/release/${RELEASE_APP_PACKAGE_VERSION}/ --overwrite --noCache`\n - URL will be printed as last output\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# Limitations / Future Development\n\n- `iframes`\n - Yeah, yeah: `iframes` are not framesets and most of the hate about iframes is probably better directed at framesets\n - The iframe serves a purpose but it stinks that it is there, primarily because it will cause issues with search bot indexing (SEO)\n - There are other options available to implement that have their own drabacks:\n - Using the `microapps-router` to proxy the \"app start\" request to a particular version of an app that then renders all of it's API resource requests to versioned URLs\n - Works only with frameworks that support hashing filenams for each deploy to unique names\n - This page would need to be marked as non-cachable\n - This may work well with Next.js which wants to know the explicit path that it will be running at (it writes that path into all resource and API requests)\n - Possible issue: the app would need to work ok being displayed at `[/{prefix}]/{appname}` when it may think that it's being displayed at `[/{prefix}]/{appname}/{semver}`\n - Disadvantage: requires some level of UI framework features (e.g. writing the absolute resource paths) to work correctly - may not work as easily for all UI frameworks\n - HTML5 added features to allow setting the relative path of all subsequent requests to be different than that displayed in the address bar\n - Gotta see if this works in modern browsers\n - Option to ditch the multiple-versions feature\n - Works only with frameworks that support hashing filenams for each deploy to unique names\n - Allows usage of the deploy and routing tooling without advantages and disadvantages of multiple-versions support\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- `microapps-publish` only supports Lambda function apps\n - There is no technical reason for the apps to only run as Lambda functions\n - Web apps could just as easily run on EC2, Kubernetes, EKS, ECS, etc\n - Anything that API Gateway can route to can work for serving a MicroApp\n - The publish tool needs to provide additional options for setting up the API Gateway route to the app\n- Authentication\n - Authentication requires rolling your own API Gateway and CloudFront deployment at the moment\n - The \"turn key\" CDK Construct should provide options to show an example of how authentication can be integrated\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- Serverless Next.js Router\n - [pwrdrvr/serverless-nextjs-router](https://github.com/pwrdrvr/serverless-nextjs-router)\n - Complementary to [@sls-next/serverless-component](https://github.com/serverless-nextjs/serverless-next.js)\n - Allows Next.js apps to run as Lambda @ Origin for speed and cost improvements vs Lambda@Edge\n - Essentially the router translates CloudFront Lambda events to API Gateway Lambda events and vice versa for responses\n - The `serverless-nextjs` project allows Next.js apps to run as Lambda functions without Express, but there was a design change to make the Lambda functions run at Edge (note: need to recheck if this changed after early 2021)\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 vs $0.0000166667 for Lambda at the origin\n - Additionally, 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 `microapps-publish`, `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 `microapps-publish` to record new app/version in the DynamoDB table, create API Gateway integrations, copy S3 assets from staging to prod bucket, etc.\n - 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/microapps-publish](https://github.com/pwrdrvr/microapps-core/tree/main/packages/microapps-publish)\n - Installed from `npm`:\n - `npm i -g @pwrdrvr/microapps-publish`\n - Node executable that updates versions in config files, deploys static assets to the S3 staging bucket, optionally compiles and deploys a new Lambda function version, and invokes `microapps-deployer`\n - 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\nNote: semi-deprecated as of 2022-01-27. Zip Lambda functions are better supported.\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 microapps-publish --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 `microapps-publish` tool.\n\n### Install Dependencies\n\n```\nnpm i --save-dev @sls-next/serverless-component@1.19.0 @pwrdrvr/serverless-nextjs-router @pwrdrvr/microapps-publish\n```\n\n### Dockerfile\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\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### deploy.json\n\nAdd this file to the root of the app.\n\nReplace `appname` with your URL path-compatible application name.\n\n```json\n{\n \"AppName\": \"appname\",\n \"SemVer\": \"0.0.0\",\n \"DefaultFile\": \"\",\n \"StaticAssetsPath\": \"./.serverless_nextjs/assets/appname/0.0.0/\",\n \"LambdaARN\": \"arn:aws:lambda:us-east-1:123456789012:function:appname:v0_0_0\",\n \"AWSAccountID\": \"123456789012\",\n \"AWSRegion\": \"us-east-2\",\n \"ServerlessNextRouterPath\": \"./node_modules/@pwrdrvr/serverless-nextjs-router/dist/index.js\"\n}\n```\n\n### serverless.yaml\n\nAdd this file to the root of the app.\n\n```yaml\nnextApp:\n component: './node_modules/@sls-next/serverless-component'\n inputs:\n deploy: false\n uploadStaticAssetsFromBuild: false\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`"
3103
+ "markdown": "![Build/Deploy CI](https://github.com/pwrdrvr/microapps-core/actions/workflows/ci.yml/badge.svg) ![Main Build](https://github.com/pwrdrvr/microapps-core/actions/workflows/main-build.yml/badge.svg) ![Release](https://github.com/pwrdrvr/microapps-core/actions/workflows/release.yml/badge.svg)\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` that looks up the version of the application to be run, then renders a transparent `iframe` with a link to that version. 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 pattern `[/{prefix}]/{appname}/{semver}/`. These \"versioned\" URLs are not intended to be advertised to end users as they would cause a user to be stuck on a particular version of the app if the URL was bookmarked. Note that the system does not limit access to particular versions of an application, as of 2022-01-26, but that can be added as a feature.\n\n# Table of Contents <!-- omit in toc -->\n\n- [Overview](#overview)\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- [Why MicroApps](#why-microapps)\n- [Limitations / Future Development](#limitations--future-development)\n- [Related Projects / Components](#related-projects--components)\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 - [deploy.json](#deployjson)\n - [serverless.yaml](#serverlessyaml)\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# 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- `cdk-sso-sync`\n - Using `npm i -g cdk-sso-sync`\n - Sets AWS SSO credentials in a way that CDK can use them\n - Not necessary if not using AWS SSO\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 microapps-publish publish -a release -n ${RELEASE_APP_PACKAGE_VERSION} -d ${DEPLOYER_LAMBDA_NAME} -l ${RELEASE_APP_LAMBDA_NAME} -s node_modules/@pwrdrvr/microapps-app-release-cdk/lib/.static_files/release/${RELEASE_APP_PACKAGE_VERSION}/ --overwrite --noCache`\n - URL will be printed as last output\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# Limitations / Future Development\n\n- `iframes`\n - Yeah, yeah: `iframes` are not framesets and most of the hate about iframes is probably better directed at framesets\n - The iframe serves a purpose but it stinks that it is there, primarily because it will cause issues with search bot indexing (SEO)\n - There are other options available to implement that have their own drabacks:\n - Using the `microapps-router` to proxy the \"app start\" request to a particular version of an app that then renders all of it's API resource requests to versioned URLs\n - Works only with frameworks that support hashing filenams for each deploy to unique names\n - This page would need to be marked as non-cachable\n - This may work well with Next.js which wants to know the explicit path that it will be running at (it writes that path into all resource and API requests)\n - Possible issue: the app would need to work ok being displayed at `[/{prefix}]/{appname}` when it may think that it's being displayed at `[/{prefix}]/{appname}/{semver}`\n - Disadvantage: requires some level of UI framework features (e.g. writing the absolute resource paths) to work correctly - may not work as easily for all UI frameworks\n - HTML5 added features to allow setting the relative path of all subsequent requests to be different than that displayed in the address bar\n - Gotta see if this works in modern browsers\n - Option to ditch the multiple-versions feature\n - Works only with frameworks that support hashing filenams for each deploy to unique names\n - Allows usage of the deploy and routing tooling without advantages and disadvantages of multiple-versions support\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- `microapps-publish` only supports Lambda function apps\n - There is no technical reason for the apps to only run as Lambda functions\n - Web apps could just as easily run on EC2, Kubernetes, EKS, ECS, etc\n - Anything that API Gateway can route to can work for serving a MicroApp\n - The publish tool needs to provide additional options for setting up the API Gateway route to the app\n- Authentication\n - Authentication requires rolling your own API Gateway and CloudFront deployment at the moment\n - The \"turn key\" CDK Construct should provide options to show an example of how authentication can be integrated\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- Serverless Next.js Router\n - [pwrdrvr/serverless-nextjs-router](https://github.com/pwrdrvr/serverless-nextjs-router)\n - Complementary to [@sls-next/serverless-component](https://github.com/serverless-nextjs/serverless-next.js)\n - Allows Next.js apps to run as Lambda @ Origin for speed and cost improvements vs Lambda@Edge\n - Essentially the router translates CloudFront Lambda events to API Gateway Lambda events and vice versa for responses\n - The `serverless-nextjs` project allows Next.js apps to run as Lambda functions without Express, but there was a design change to make the Lambda functions run at Edge (note: need to recheck if this changed after early 2021)\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 vs $0.0000166667 for Lambda at the origin\n - Additionally, 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 `microapps-publish`, `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 `microapps-publish` to record new app/version in the DynamoDB table, create API Gateway integrations, copy S3 assets from staging to prod bucket, etc.\n - 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/microapps-publish](https://github.com/pwrdrvr/microapps-core/tree/main/packages/microapps-publish)\n - Installed from `npm`:\n - `npm i -g @pwrdrvr/microapps-publish`\n - Node executable that updates versions in config files, deploys static assets to the S3 staging bucket, optionally compiles and deploys a new Lambda function version, and invokes `microapps-deployer`\n - 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\nNote: semi-deprecated as of 2022-01-27. Zip Lambda functions are better supported.\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 microapps-publish --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 `microapps-publish` tool.\n\n### Install Dependencies\n\n```\nnpm i --save-dev @sls-next/serverless-component@1.19.0 @pwrdrvr/serverless-nextjs-router @pwrdrvr/microapps-publish\n```\n\n### Dockerfile\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\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### deploy.json\n\nAdd this file to the root of the app.\n\nReplace `appname` with your URL path-compatible application name.\n\n```json\n{\n \"AppName\": \"appname\",\n \"SemVer\": \"0.0.0\",\n \"DefaultFile\": \"\",\n \"StaticAssetsPath\": \"./.serverless_nextjs/assets/appname/0.0.0/\",\n \"LambdaARN\": \"arn:aws:lambda:us-east-1:123456789012:function:appname:v0_0_0\",\n \"AWSAccountID\": \"123456789012\",\n \"AWSRegion\": \"us-east-2\",\n \"ServerlessNextRouterPath\": \"./node_modules/@pwrdrvr/serverless-nextjs-router/dist/index.js\"\n}\n```\n\n### serverless.yaml\n\nAdd this file to the root of the app.\n\n```yaml\nnextApp:\n component: './node_modules/@sls-next/serverless-component'\n inputs:\n deploy: false\n uploadStaticAssetsFromBuild: false\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`"
3104
3104
  },
3105
3105
  "repository": {
3106
3106
  "type": "git",
@@ -3371,7 +3371,7 @@
3371
3371
  "kind": "interface",
3372
3372
  "locationInModule": {
3373
3373
  "filename": "src/MicroAppsEdgeToOrigin.ts",
3374
- "line": 102
3374
+ "line": 103
3375
3375
  },
3376
3376
  "name": "GenerateEdgeToOriginConfigOptions",
3377
3377
  "properties": [
@@ -3383,7 +3383,7 @@
3383
3383
  "immutable": true,
3384
3384
  "locationInModule": {
3385
3385
  "filename": "src/MicroAppsEdgeToOrigin.ts",
3386
- "line": 105
3386
+ "line": 106
3387
3387
  },
3388
3388
  "name": "addXForwardedHostHeader",
3389
3389
  "type": {
@@ -3398,7 +3398,7 @@
3398
3398
  "immutable": true,
3399
3399
  "locationInModule": {
3400
3400
  "filename": "src/MicroAppsEdgeToOrigin.ts",
3401
- "line": 103
3401
+ "line": 104
3402
3402
  },
3403
3403
  "name": "originRegion",
3404
3404
  "type": {
@@ -3413,7 +3413,7 @@
3413
3413
  "immutable": true,
3414
3414
  "locationInModule": {
3415
3415
  "filename": "src/MicroAppsEdgeToOrigin.ts",
3416
- "line": 106
3416
+ "line": 107
3417
3417
  },
3418
3418
  "name": "replaceHostHeader",
3419
3419
  "type": {
@@ -3428,7 +3428,7 @@
3428
3428
  "immutable": true,
3429
3429
  "locationInModule": {
3430
3430
  "filename": "src/MicroAppsEdgeToOrigin.ts",
3431
- "line": 104
3431
+ "line": 105
3432
3432
  },
3433
3433
  "name": "signingMode",
3434
3434
  "type": {
@@ -3629,7 +3629,7 @@
3629
3629
  "kind": "interface",
3630
3630
  "locationInModule": {
3631
3631
  "filename": "src/MicroAppsEdgeToOrigin.ts",
3632
- "line": 15
3632
+ "line": 16
3633
3633
  },
3634
3634
  "name": "IMicroAppsEdgeToOrigin",
3635
3635
  "properties": [
@@ -3643,7 +3643,7 @@
3643
3643
  "immutable": true,
3644
3644
  "locationInModule": {
3645
3645
  "filename": "src/MicroAppsEdgeToOrigin.ts",
3646
- "line": 21
3646
+ "line": 22
3647
3647
  },
3648
3648
  "name": "edgeToOriginFunction",
3649
3649
  "type": {
@@ -3668,7 +3668,7 @@
3668
3668
  "immutable": true,
3669
3669
  "locationInModule": {
3670
3670
  "filename": "src/MicroAppsEdgeToOrigin.ts",
3671
- "line": 26
3671
+ "line": 27
3672
3672
  },
3673
3673
  "name": "edgeToOriginLambdas",
3674
3674
  "type": {
@@ -4700,7 +4700,7 @@
4700
4700
  },
4701
4701
  "locationInModule": {
4702
4702
  "filename": "src/MicroAppsEdgeToOrigin.ts",
4703
- "line": 135
4703
+ "line": 136
4704
4704
  },
4705
4705
  "parameters": [
4706
4706
  {
@@ -4729,7 +4729,7 @@
4729
4729
  "kind": "class",
4730
4730
  "locationInModule": {
4731
4731
  "filename": "src/MicroAppsEdgeToOrigin.ts",
4732
- "line": 112
4732
+ "line": 113
4733
4733
  },
4734
4734
  "methods": [
4735
4735
  {
@@ -4740,7 +4740,7 @@
4740
4740
  },
4741
4741
  "locationInModule": {
4742
4742
  "filename": "src/MicroAppsEdgeToOrigin.ts",
4743
- "line": 118
4743
+ "line": 119
4744
4744
  },
4745
4745
  "name": "generateEdgeToOriginConfig",
4746
4746
  "parameters": [
@@ -4770,7 +4770,7 @@
4770
4770
  "immutable": true,
4771
4771
  "locationInModule": {
4772
4772
  "filename": "src/MicroAppsEdgeToOrigin.ts",
4773
- "line": 126
4773
+ "line": 127
4774
4774
  },
4775
4775
  "name": "edgeToOriginFunction",
4776
4776
  "overrides": "@pwrdrvr/microapps-cdk.IMicroAppsEdgeToOrigin",
@@ -4795,7 +4795,7 @@
4795
4795
  "immutable": true,
4796
4796
  "locationInModule": {
4797
4797
  "filename": "src/MicroAppsEdgeToOrigin.ts",
4798
- "line": 131
4798
+ "line": 132
4799
4799
  },
4800
4800
  "name": "edgeToOriginLambdas",
4801
4801
  "overrides": "@pwrdrvr/microapps-cdk.IMicroAppsEdgeToOrigin",
@@ -4822,7 +4822,7 @@
4822
4822
  "kind": "interface",
4823
4823
  "locationInModule": {
4824
4824
  "filename": "src/MicroAppsEdgeToOrigin.ts",
4825
- "line": 32
4825
+ "line": 33
4826
4826
  },
4827
4827
  "name": "MicroAppsEdgeToOriginProps",
4828
4828
  "properties": [
@@ -4837,7 +4837,7 @@
4837
4837
  "immutable": true,
4838
4838
  "locationInModule": {
4839
4839
  "filename": "src/MicroAppsEdgeToOrigin.ts",
4840
- "line": 65
4840
+ "line": 66
4841
4841
  },
4842
4842
  "name": "addXForwardedHostHeader",
4843
4843
  "optional": true,
@@ -4856,7 +4856,7 @@
4856
4856
  "immutable": true,
4857
4857
  "locationInModule": {
4858
4858
  "filename": "src/MicroAppsEdgeToOrigin.ts",
4859
- "line": 46
4859
+ "line": 47
4860
4860
  },
4861
4861
  "name": "assetNameRoot",
4862
4862
  "optional": true,
@@ -4875,7 +4875,7 @@
4875
4875
  "immutable": true,
4876
4876
  "locationInModule": {
4877
4877
  "filename": "src/MicroAppsEdgeToOrigin.ts",
4878
- "line": 54
4878
+ "line": 55
4879
4879
  },
4880
4880
  "name": "assetNameSuffix",
4881
4881
  "optional": true,
@@ -4893,7 +4893,7 @@
4893
4893
  "immutable": true,
4894
4894
  "locationInModule": {
4895
4895
  "filename": "src/MicroAppsEdgeToOrigin.ts",
4896
- "line": 99
4896
+ "line": 100
4897
4897
  },
4898
4898
  "name": "originRegion",
4899
4899
  "optional": true,
@@ -4911,7 +4911,7 @@
4911
4911
  "immutable": true,
4912
4912
  "locationInModule": {
4913
4913
  "filename": "src/MicroAppsEdgeToOrigin.ts",
4914
- "line": 38
4914
+ "line": 39
4915
4915
  },
4916
4916
  "name": "removalPolicy",
4917
4917
  "optional": true,
@@ -4930,7 +4930,7 @@
4930
4930
  "immutable": true,
4931
4931
  "locationInModule": {
4932
4932
  "filename": "src/MicroAppsEdgeToOrigin.ts",
4933
- "line": 77
4933
+ "line": 78
4934
4934
  },
4935
4935
  "name": "replaceHostHeader",
4936
4936
  "optional": true,
@@ -4949,7 +4949,7 @@
4949
4949
  "immutable": true,
4950
4950
  "locationInModule": {
4951
4951
  "filename": "src/MicroAppsEdgeToOrigin.ts",
4952
- "line": 90
4952
+ "line": 91
4953
4953
  },
4954
4954
  "name": "signingMode",
4955
4955
  "optional": true,
@@ -6163,6 +6163,6 @@
6163
6163
  "symbolId": "src/MicroAppsTable:MicroAppsTableProps"
6164
6164
  }
6165
6165
  },
6166
- "version": "0.2.12",
6167
- "fingerprint": "e/SGzClAEVnA05j9b8maqC+wYjffg5sOd16is2KSWIM="
6166
+ "version": "0.2.13",
6167
+ "fingerprint": "3fP6FV20Rxpk08/TMT9o63oJ0V+VdhpD8QzDWMMWjuY="
6168
6168
  }
package/README.md CHANGED
@@ -68,7 +68,7 @@ For development / testing purposes only, each version of an applicaton can be ac
68
68
  - `./deploy.sh`
69
69
  - Deploys the CDK Stack
70
70
  - Essentially runs two commands along with extraction of outputs:
71
- - `npx cdk deploy --context @pwrdrvr/microapps:stackName=microapps-demo-deploy --context @pwrdrvr/microapps:deployReleaseApp=true microapps-basic`
71
+ - `npx cdk deploy --context @pwrdrvr/microapps:deployReleaseApp=true microapps-basic`
72
72
  - `npx microapps-publish publish -a release -n ${RELEASE_APP_PACKAGE_VERSION} -d ${DEPLOYER_LAMBDA_NAME} -l ${RELEASE_APP_LAMBDA_NAME} -s node_modules/@pwrdrvr/microapps-app-release-cdk/lib/.static_files/release/${RELEASE_APP_PACKAGE_VERSION}/ --overwrite --noCache`
73
73
  - URL will be printed as last output
74
74
 
package/changelog.md CHANGED
@@ -1,2 +1,2 @@
1
1
 
2
- ### [0.2.12](https://github.com/pwrdrvr/microapps-core/compare/v0.2.11...v0.2.12) (2022-05-23)
2
+ ### [0.2.13](https://github.com/pwrdrvr/microapps-core/compare/v0.2.12...v0.2.13) (2022-10-05)
package/lib/MicroApps.js CHANGED
@@ -119,5 +119,5 @@ class MicroApps extends constructs_1.Construct {
119
119
  }
120
120
  exports.MicroApps = MicroApps;
121
121
  _a = JSII_RTTI_SYMBOL_1;
122
- MicroApps[_a] = { fqn: "@pwrdrvr/microapps-cdk.MicroApps", version: "0.2.12" };
122
+ MicroApps[_a] = { fqn: "@pwrdrvr/microapps-cdk.MicroApps", version: "0.2.13" };
123
123
  //# 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 dynamodb from 'aws-cdk-lib/aws-dynamodb';\nimport * as r53 from 'aws-cdk-lib/aws-route53';\nimport { Construct } from 'constructs';\nimport { IMicroAppsAPIGwy, MicroAppsAPIGwy } from './MicroAppsAPIGwy';\nimport { IMicroAppsCF, MicroAppsCF } from './MicroAppsCF';\nimport { IMicroAppsEdgeToOrigin, MicroAppsEdgeToOrigin } from './MicroAppsEdgeToOrigin';\nimport { IMicroAppsS3, MicroAppsS3 } from './MicroAppsS3';\nimport { IMicroAppsSvcs, MicroAppsSvcs } from './MicroAppsSvcs';\nimport { reverseDomain } from './utils/ReverseDomain';\n\n/**\n * A CDK Construct for creating a MicroApps runtime environment used\n * to host Next.js, React, or any other sort of web application with\n * multiple versions available for comparison, quick rollbacks, quick\n * releases, and a complete lack of user disturbance on deploys.\n *\n * @remarks\n *\n * {@link MicroApps} provides a turn-key construct that creates all\n * dependencies with limited exposure of underlying AWS Resource options.\n * This construct is the easiest to use when exploring MicroApps for the\n * first time.\n *\n * {@link MicroAppsAPIGwy}, {@link MicroAppsCF}, {@link MicroAppsS3},\n * and {@link MicroAppsSvcs}, and their helper static methods, can be used\n * to create AWS Resources more directly, to provide your own AWS Resources\n * (e.g. an existing CloudFront Distribution), and to have more flexibility\n * than the {@link MicroApps} construct offers.\n *\n * @packageDocumentation\n */\n\n/**\n * Properties to initialize an instance of `MicroApps`.\n */\nexport interface MicroAppsProps {\n  /**\n   * RemovalPolicy override for child resources\n   *\n   * Note: if set to DESTROY the S3 buckes will have `autoDeleteObjects` set to `true`\n   *\n   * @default - per resource default\n   */\n  readonly removalPolicy?: RemovalPolicy;\n\n  /**\n   * Passed to NODE_ENV of Router and Deployer Lambda functions.\n   *\n   * @default dev\n   */\n  readonly appEnv: string;\n\n  /**\n   * Optional asset name root\n   *\n   * @example microapps\n   * @default - resource names auto assigned\n   */\n  readonly assetNameRoot?: string;\n\n  /**\n   * Optional asset name suffix\n   *\n   * @example -dev-pr-12\n   * @default none\n   */\n  readonly assetNameSuffix?: string;\n\n  /**\n   * Route53 zone in which to create optional `domainNameEdge` record\n   */\n  readonly r53Zone?: r53.IHostedZone;\n\n  /**\n   * Certificate in US-East-1 for the CloudFront distribution.\n   */\n  readonly certEdge?: acm.ICertificate;\n\n  /**\n   * Certificate in deployed region for the API Gateway.\n   */\n  readonly certOrigin?: acm.ICertificate;\n\n  /**\n   * Use a strict S3 Bucket Policy that prevents applications\n   * from reading/writing/modifying/deleting files in the S3 Bucket\n   * outside of the path that is specific to their app/version.\n   *\n   * This setting should be used when applications are less than\n   * fully trusted.\n   *\n   * @default false\n   */\n  readonly s3StrictBucketPolicy?: boolean;\n\n  /**\n   * Applies when using s3StrictBucketPolicy = true\n   *\n   * IAM Role or IAM User names to exclude from the DENY rules on the S3 Bucket Policy.\n   *\n   * Roles that are Assumed must instead have their AROA added to `s3PolicyBypassAROAs`.\n   *\n   * Typically any admin roles / users that need to view or manage the S3 Bucket\n   * would be added to this list.\n   *\n   * @example ['arn:aws:iam::1234567890123:role/AdminAccess', 'arn:aws:iam::1234567890123:user/MyAdminUser']\n   *\n   * @see s3PolicyBypassAROAs\n   */\n  readonly s3PolicyBypassPrincipalARNs?: string[];\n\n  /**\n   * Applies when using s3StrictBucketPolicy = true\n   *\n   * AROAs of the IAM Role to exclude from the DENY rules on the S3 Bucket Policy.\n   * This allows sessions that assume the IAM Role to be excluded from the\n   * DENY rules on the S3 Bucket Policy.\n   *\n   * Typically any admin roles / users that need to view or manage the S3 Bucket\n   * would be added to this list.\n   *\n   * Roles / users that are used directly, not assumed, can be added to `s3PolicyBypassRoleNames` instead.\n   *\n   * Note: This AROA must be specified to prevent this policy from locking\n   * out non-root sessions that have assumed the admin role.\n   *\n   * The notPrincipals will only match the role name exactly and will not match\n   * any session that has assumed the role since notPrincipals does not allow\n   * wildcard matches and does not do wildcard matches implicitly either.\n   *\n   * The AROA must be used because there are only 3 Principal variables available:\n   *  https://docs.aws.amazon.com/IAM/latest/UserGuide/reference_policies_variables.html#principaltable\n   *  aws:username, aws:userid, aws:PrincipalTag\n   *\n   * For an assumed role, aws:username is blank, aws:userid is:\n   *  [unique id AKA AROA for Role]:[session name]\n   *\n   * Table of unique ID prefixes such as AROA:\n   *  https://docs.aws.amazon.com/IAM/latest/UserGuide/reference_identifiers.html#identifiers-prefixes\n   *\n   * The name of the role is simply not available for an assumed role and, if it was,\n   * a complicated comparison would be requierd to prevent exclusion\n   * of applying the Deny Rule to roles from other accounts.\n   *\n   * To get the AROA with the AWS CLI:\n   *   aws iam get-role --role-name ROLE-NAME\n   *   aws iam get-user -–user-name USER-NAME\n   *\n   * @example [ 'AROA1234567890123' ]\n   *\n   * @see s3StrictBucketPolicy\n   */\n  readonly s3PolicyBypassAROAs?: string[];\n\n  /**\n   * Optional custom domain name for the CloudFront distribution.\n   *\n   * @example apps.pwrdrvr.com\n   * @default auto-assigned\n   */\n  readonly domainNameEdge?: string;\n\n  /**\n   * Optional custom domain name for the API Gateway HTTPv2 API.\n   *\n   * @example apps-origin.pwrdrvr.com\n   * @default auto-assigned\n   */\n  readonly domainNameOrigin?: string;\n\n  /**\n   * Path prefix on the root of the CloudFront distribution\n   *\n   * @example dev/\n   */\n  readonly rootPathPrefix?: string;\n\n  /**\n   * Create 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   * 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 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   * Existing table for apps/versions/rules\n   *\n   * @warning - It is *strongly* suggested that production stacks create\n   * their own DynamoDB Table and pass it into this construct, for protection\n   * against data loss due to logical ID changes, the ability to configure\n   * Provisioned capacity with Auto Scaling, the ability to add additional indices, etc.\n   *\n   * Requirements:\n   * - Hash Key: `PK`\n   * - Sort Key: `SK`\n   *\n   * @default created by construct\n   */\n  readonly table?: dynamodb.ITable;\n}\n\n/**\n * Represents a MicroApps\n */\nexport interface IMicroApps {\n  /** {@inheritdoc IMicroAppsCF} */\n  readonly cf: IMicroAppsCF;\n\n  /** {@inheritdoc IMicroAppsEdgeToOrigin} */\n  readonly edgeToOrigin?: IMicroAppsEdgeToOrigin;\n\n  /** {@inheritdoc IMicroAppsS3} */\n  readonly s3: IMicroAppsS3;\n\n  /** {@inheritdoc IMicroAppsSvcs} */\n  readonly svcs: IMicroAppsSvcs;\n\n  /** {@inheritdoc IMicroAppsAPIGwy} */\n  readonly apigwy: IMicroAppsAPIGwy;\n}\n\n/**\n * Create a new MicroApps \"turnkey\" construct for simple\n * deployments and for initial evaulation of the MicroApps framework.\n *\n * Use this construct to create a PoC working entire stack.\n *\n * Do not use this construct when adding MicroApps to an existing\n * CloudFront, API Gateway, S3 Bucket, etc. or where access\n * to all features of the AWS Resources are needed (e.g. to\n * add additional Behaviors to the CloudFront distribution, set authorizors\n * on API Gateway, etc.).\n *\n * @warning This construct is not intended for production use.\n * In a production stack the DynamoDB Table, API Gateway, S3 Buckets,\n * etc. should be created in a \"durable\" stack where the IDs will not\n * change and where changes to the MicroApps construct will not\n * cause failures to deploy or data to be deleted.\n *\n *  @see {@link https://github.com/pwrdrvr/microapps-core/blob/main/packages/cdk/lib/MicroApps.ts | example usage in a CDK Stack }\n */\nexport class MicroApps extends Construct implements IMicroApps {\n  private _cf: MicroAppsCF;\n  public get cf(): IMicroAppsCF {\n    return this._cf;\n  }\n\n  private _edgeToOrigin?: MicroAppsEdgeToOrigin;\n  public get edgeToOrigin(): IMicroAppsEdgeToOrigin | undefined {\n    return this._edgeToOrigin;\n  }\n\n  private _s3: MicroAppsS3;\n  public get s3(): IMicroAppsS3 {\n    return this._s3;\n  }\n\n  private _apigwy: MicroAppsAPIGwy;\n  public get apigwy(): IMicroAppsAPIGwy {\n    return this._apigwy;\n  }\n\n  private _svcs: MicroAppsSvcs;\n  public get svcs(): IMicroAppsSvcs {\n    return this._svcs;\n  }\n\n  constructor(scope: Construct, id: string, props?: MicroAppsProps) {\n    super(scope, id);\n\n    if (props === undefined) {\n      throw new Error('props must be set');\n    }\n\n    const {\n      domainNameEdge,\n      domainNameOrigin,\n      assetNameRoot,\n      assetNameSuffix,\n      r53Zone,\n      certEdge,\n      appEnv = 'dev',\n      certOrigin,\n      removalPolicy,\n      s3PolicyBypassAROAs,\n      s3PolicyBypassPrincipalARNs,\n      s3StrictBucketPolicy,\n      rootPathPrefix,\n      createAPIPathRoute = true,\n      addXForwardedHostHeader = true,\n      replaceHostHeader = true,\n      signingMode = 'sign',\n      originRegion,\n      table,\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    });\n    this._apigwy = new MicroAppsAPIGwy(this, 'api', {\n      removalPolicy,\n      assetNameRoot,\n      assetNameSuffix,\n      domainNameEdge,\n      domainNameOrigin,\n      r53Zone,\n      certOrigin,\n      rootPathPrefix,\n      requireIAMAuthorization: signingMode !== 'none',\n    });\n    if (signingMode !== 'none' || replaceHostHeader || addXForwardedHostHeader) {\n      this._edgeToOrigin = new MicroAppsEdgeToOrigin(this, 'edgeToOrigin', {\n        assetNameRoot,\n        assetNameSuffix,\n        removalPolicy,\n        addXForwardedHostHeader,\n        replaceHostHeader,\n        originRegion,\n        signingMode,\n      });\n    }\n    this._cf = new MicroAppsCF(this, 'cft', {\n      removalPolicy,\n      assetNameRoot,\n      assetNameSuffix,\n      domainNameEdge,\n      domainNameOrigin,\n      httpApi: this._apigwy.httpApi,\n      r53Zone,\n      certEdge,\n      bucketAppsOrigin: this._s3.bucketAppsOrigin,\n      bucketLogs: this._s3.bucketLogs,\n      rootPathPrefix,\n      createAPIPathRoute,\n      edgeToOriginLambdas: this._edgeToOrigin ? this._edgeToOrigin.edgeToOriginLambdas : undefined,\n    });\n    this._svcs = new MicroAppsSvcs(this, 'svcs', {\n      httpApi: this.apigwy.httpApi,\n      removalPolicy,\n      bucketApps: this._s3.bucketApps,\n      bucketAppsOAI: this._s3.bucketAppsOAI,\n      bucketAppsStaging: this._s3.bucketAppsStaging,\n      assetNameRoot,\n      assetNameSuffix,\n      appEnv,\n      s3PolicyBypassAROAs,\n      s3PolicyBypassPrincipalARNs,\n      s3StrictBucketPolicy,\n      rootPathPrefix,\n      requireIAMAuthorization: signingMode !== 'none',\n      table,\n    });\n  }\n}\n"]}
@@ -167,5 +167,5 @@ class MicroAppsAPIGwy extends constructs_1.Construct {
167
167
  }
168
168
  exports.MicroAppsAPIGwy = MicroAppsAPIGwy;
169
169
  _a = JSII_RTTI_SYMBOL_1;
170
- MicroAppsAPIGwy[_a] = { fqn: "@pwrdrvr/microapps-cdk.MicroAppsAPIGwy", version: "0.2.12" };
170
+ MicroAppsAPIGwy[_a] = { fqn: "@pwrdrvr/microapps-cdk.MicroAppsAPIGwy", version: "0.2.13" };
171
171
  //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"MicroAppsAPIGwy.js","sourceRoot":"","sources":["../src/MicroAppsAPIGwy.ts"],"names":[],"mappings":"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","sourcesContent":["import * as apigwy from '@aws-cdk/aws-apigatewayv2-alpha';\nimport * as apigwyAuth from '@aws-cdk/aws-apigatewayv2-authorizers-alpha';\nimport { RemovalPolicy, Stack, Tags } from 'aws-cdk-lib';\nimport * as apigwycfn from 'aws-cdk-lib/aws-apigatewayv2';\nimport * as acm from 'aws-cdk-lib/aws-certificatemanager';\nimport * as iam from 'aws-cdk-lib/aws-iam';\nimport * as logs from 'aws-cdk-lib/aws-logs';\nimport * as r53 from 'aws-cdk-lib/aws-route53';\nimport * as r53targets from 'aws-cdk-lib/aws-route53-targets';\nimport { Construct } from 'constructs';\n\n/**\n * Properties to initialize an instance of `MicroAppsAPIGwy`.\n */\nexport interface MicroAppsAPIGwyProps {\n  /**\n   * RemovalPolicy override for child resources\n   *\n   * Note: if set to DESTROY the S3 buckes will have `autoDeleteObjects` set to `true`\n   *\n   * @default - per resource default\n   */\n  readonly removalPolicy?: RemovalPolicy;\n\n  /**\n   * CloudFront edge domain name\n   *\n   * @example apps.pwrdrvr.com\n   * @default auto-assigned\n   */\n  readonly domainNameEdge?: string;\n\n  /**\n   * API Gateway origin domain name\n   *\n   * @example apps-origin.pwrdrvr.com\n   * @default auto-assigned\n   */\n  readonly domainNameOrigin?: string;\n\n  /**\n   * Optional asset name root\n   *\n   * @example microapps\n   * @default - resource names auto assigned\n   */\n  readonly assetNameRoot?: string;\n\n  /**\n   * Optional asset name suffix\n   *\n   * @example -dev-pr-12\n   * @default none\n   */\n  readonly assetNameSuffix?: string;\n\n  /**\n   * Optional local region ACM certificate to use for API Gateway\n   * Note: required when using a custom domain\n   *\n   * @default none\n   */\n  readonly certOrigin?: acm.ICertificate;\n\n  /**\n   * Route53 zone in which to create optional `domainNameEdge` record\n   */\n  readonly r53Zone?: r53.IHostedZone;\n\n  /**\n   * Path prefix on the root of the API Gateway Stage\n   *\n   * @example dev/\n   * @default none\n   */\n  readonly rootPathPrefix?: string;\n\n  /**\n   * Require IAM auth on API Gateway\n   *\n   * @default true\n   */\n  readonly requireIAMAuthorization?: boolean;\n}\n\n/**\n * Represents a MicroApps API Gateway\n */\nexport interface IMicroAppsAPIGwy {\n  /**\n   * Domain Name applied to API Gateway origin\n   */\n  readonly dnAppsOrigin?: apigwy.IDomainName;\n\n  /**\n   * API Gateway\n   */\n  readonly httpApi: apigwy.HttpApi;\n}\n\n/**\n * Create a new MicroApps API Gateway HTTP API endpoint, optionally\n * requiring IAM authorization\n */\nexport class MicroAppsAPIGwy extends Construct implements IMicroAppsAPIGwy {\n  private _dnAppsOrigin: apigwy.DomainName | undefined;\n  public get dnAppsOrigin(): apigwy.IDomainName | undefined {\n    return this._dnAppsOrigin;\n  }\n\n  private _httpApi: apigwy.HttpApi;\n  public get httpApi(): apigwy.HttpApi {\n    return this._httpApi;\n  }\n\n  constructor(scope: Construct, id: string, props: MicroAppsAPIGwyProps) {\n    super(scope, id);\n\n    if (props === undefined) {\n      throw new Error('props cannot be undefined');\n    }\n\n    if (\n      (props.r53Zone === undefined && props.domainNameEdge !== undefined) ||\n      (props.r53Zone !== undefined && props.domainNameEdge === undefined)\n    ) {\n      throw new Error('If either of r53Zone or domainNameEdge are set then the other must be set');\n    }\n\n    if (props.domainNameOrigin !== undefined && props.certOrigin === undefined) {\n      throw new Error('If domainNameOrigin is set then certOrigin must be set');\n    }\n\n    if (\n      (props.domainNameEdge === undefined && props.certOrigin !== undefined) ||\n      (props.domainNameEdge !== undefined && props.certOrigin === undefined)\n    ) {\n      throw new Error(\n        'If either of domainNameEdge or certOrigin are set then the other must be set',\n      );\n    }\n\n    const {\n      r53Zone,\n      domainNameEdge,\n      domainNameOrigin,\n      certOrigin,\n      removalPolicy,\n      assetNameRoot,\n      assetNameSuffix,\n      rootPathPrefix,\n      requireIAMAuthorization = true,\n    } = props;\n\n    // API Gateway uses the `id` string as the gateway name without\n    // any randomization... we have to make sure the name is unique-ish\n    const apigatewayName = assetNameRoot\n      ? `${assetNameRoot}${assetNameSuffix}`\n      : `${Stack.of(this).stackName}-microapps`;\n\n    //\n    // APIGateway domain names for CloudFront and origin\n    //\n    this._httpApi = new apigwy.HttpApi(this, 'gwy', {\n      apiName: apigatewayName,\n      createDefaultStage: false,\n      defaultAuthorizer: requireIAMAuthorization ? new apigwyAuth.HttpIamAuthorizer() : undefined,\n    });\n    if (removalPolicy !== undefined) {\n      this._httpApi.applyRemovalPolicy(removalPolicy);\n    }\n    // This allows the Lambda @ Edge function to execute this api\n    Tags.of(this._httpApi).add('microapp-managed', 'true');\n\n    // Create the stage\n    const stage = new apigwy.HttpStage(this, 'stage', {\n      httpApi: this._httpApi,\n      autoDeploy: true,\n      // If rootPathPrefix is not defined this will be the $default stage\n      stageName: rootPathPrefix,\n    });\n    // This allows the Lambda @ Edge function to execute this api\n    Tags.of(stage).add('microapp-managed', 'true');\n\n    if (domainNameEdge !== undefined && certOrigin !== undefined) {\n      // Create Custom Domains for API Gateway\n      const dnAppsEdge = new apigwy.DomainName(this, 'microapps-apps-edge-dn', {\n        domainName: domainNameEdge,\n        certificate: certOrigin,\n      });\n      if (removalPolicy !== undefined) {\n        dnAppsEdge.applyRemovalPolicy(removalPolicy);\n      }\n\n      // Create the edge domain name mapping for the API\n      const apiMapping = new apigwy.ApiMapping(this, 'mapping', {\n        api: this._httpApi,\n        domainName: dnAppsEdge,\n        stage,\n      });\n      // 2022-01-16 - CDK is still generating CloudFormation with no dependency\n      // between the R53 RecordSet and the Mapping\n      apiMapping.node.addDependency(dnAppsEdge);\n    }\n\n    if (domainNameOrigin !== undefined && certOrigin !== undefined) {\n      this._dnAppsOrigin = new apigwy.DomainName(this, 'origin-dn', {\n        domainName: domainNameOrigin,\n        certificate: certOrigin,\n      });\n      if (removalPolicy !== undefined) {\n        this._dnAppsOrigin.applyRemovalPolicy(removalPolicy);\n      }\n    }\n\n    // Enable access logs on API Gateway\n    const apiAccessLogs = new logs.LogGroup(this, 'logs', {\n      logGroupName: apigatewayName\n        ? `/aws/apigwy/${apigatewayName}`\n        : `/aws/apigwy/${this.httpApi.httpApiName}`,\n      retention: logs.RetentionDays.TWO_WEEKS,\n    });\n    if (removalPolicy !== undefined) {\n      apiAccessLogs.applyRemovalPolicy(removalPolicy);\n    }\n    const cfnStage = stage.node.defaultChild as apigwycfn.CfnStage;\n    cfnStage.accessLogSettings = {\n      destinationArn: apiAccessLogs.logGroupArn,\n      format: JSON.stringify({\n        requestId: '$context.requestId',\n        userAgent: '$context.identity.userAgent',\n        sourceIp: '$context.identity.sourceIp',\n        requestTime: '$context.requestTime',\n        requestTimeEpoch: '$context.requestTimeEpoch',\n        httpMethod: '$context.httpMethod',\n        path: '$context.path',\n        status: '$context.status',\n        protocol: '$context.protocol',\n        responseLength: '$context.responseLength',\n        domainName: '$context.domainName',\n        authorizerError: '$context.authorizer.error',\n      }),\n    };\n\n    // Create a logging role\n    // Tips: https://github.com/aws/aws-cdk/issues/11100\n    const apiGwyLogRole = new iam.Role(this, 'logs-role', {\n      managedPolicies: [\n        iam.ManagedPolicy.fromAwsManagedPolicyName(\n          'service-role/AmazonAPIGatewayPushToCloudWatchLogs',\n        ),\n      ],\n      assumedBy: new iam.ServicePrincipal('apigateway.amazonaws.com'),\n    });\n    apiAccessLogs.grantWrite(apiGwyLogRole);\n\n    //\n    // Let API Gateway accept requests using domainNameOrigin\n    // That is the origin URI that CloudFront uses for this gateway.\n    // The gateway will refuse the traffic if it doesn't have the\n    // domain name registered.\n    //\n    if (this._dnAppsOrigin !== undefined) {\n      const mappingAppsApis = new apigwy.ApiMapping(this, 'api-map-origin', {\n        api: this._httpApi,\n        domainName: this._dnAppsOrigin,\n        stage,\n      });\n      // 2022-01-16 - CDK is still generating CloudFormation with no dependency\n      // between the R53 RecordSet and the Mapping\n      mappingAppsApis.node.addDependency(this._dnAppsOrigin);\n      if (removalPolicy !== undefined) {\n        mappingAppsApis.applyRemovalPolicy(removalPolicy);\n      }\n    }\n\n    //\n    // Create the origin name for API Gateway\n    //\n    if (r53Zone !== undefined && this._dnAppsOrigin) {\n      const rrAppsOrigin = new r53.ARecord(this, 'origin-arecord', {\n        zone: r53Zone,\n        recordName: domainNameOrigin,\n        target: r53.RecordTarget.fromAlias(\n          new r53targets.ApiGatewayv2DomainProperties(\n            this._dnAppsOrigin.regionalDomainName,\n            this._dnAppsOrigin.regionalHostedZoneId,\n          ),\n        ),\n      });\n      if (removalPolicy !== undefined) {\n        rrAppsOrigin.applyRemovalPolicy(removalPolicy);\n      }\n    }\n  }\n}\n"]}
@@ -205,5 +205,5 @@ class MicroAppsCF extends constructs_1.Construct {
205
205
  }
206
206
  exports.MicroAppsCF = MicroAppsCF;
207
207
  _a = JSII_RTTI_SYMBOL_1;
208
- MicroAppsCF[_a] = { fqn: "@pwrdrvr/microapps-cdk.MicroAppsCF", version: "0.2.12" };
208
+ MicroAppsCF[_a] = { fqn: "@pwrdrvr/microapps-cdk.MicroAppsCF", version: "0.2.13" };
209
209
  //# 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 * as apigwy from '@aws-cdk/aws-apigatewayv2-alpha';\nimport { Aws, 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   */\n  readonly bucketAppsOrigin: 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   * API Gateway v2 HTTP API for apps\n   */\n  readonly httpApi: apigwy.HttpApi;\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\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   * Configuration of the edge to origin lambda functions\n   *\n   * @defaunt - no edge to API Gateway origin functions added\n   */\n  readonly edgeToOriginLambdas?: cf.EdgeLambda[];\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   * API Gateway CloudFront Origin for API calls\n   */\n  readonly apiGwyOrigin: cf.IOrigin;\n\n  /**\n   * S3 Bucket CloudFront Origin for static assets\n   */\n  readonly bucketAppsOrigin: cforigins.S3Origin;\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 apigwyOriginRequestPolicy: 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   * 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   * Edge lambdas to associate with the API Gateway routes\n   */\n  readonly apigwyEdgeFunctions?: 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      apiGwyOrigin,\n      bucketAppsOrigin,\n      distro,\n      apigwyOriginRequestPolicy,\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.CORS_S3_ORIGIN,\n      viewerProtocolPolicy: cf.ViewerProtocolPolicy.REDIRECT_TO_HTTPS,\n    };\n    const apiGwyBehaviorOptions: 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: apigwyOriginRequestPolicy,\n      viewerProtocolPolicy: cf.ViewerProtocolPolicy.REDIRECT_TO_HTTPS,\n      edgeLambdas: props.apigwyEdgeFunctions,\n    };\n\n    //\n    // If a route specifically has `/api/` in it, send it to API Gateway\n    // This is needed to catch routes that have periods in the API path data,\n    // such as: /release/0.0.0/api/update/default/release/0.0.0\n    //\n    if (createAPIPathRoute) {\n      distro.addBehavior(\n        posixPath.join(rootPathPrefix, '/*/*/api/*'),\n        apiGwyOrigin,\n        apiGwyBehaviorOptions,\n      );\n    }\n\n    //\n    // If a route specifically has `/_next/data/` in it, send it to API Gateway\n    // This is needed to catch routes that have periods in the API path data,\n    // such as: /release/0.0.0/_next/data/app.json\n    //\n    if (createNextDataPathRoute) {\n      distro.addBehavior(\n        posixPath.join(rootPathPrefix, '/*/*/_next/data/*'),\n        apiGwyOrigin,\n        apiGwyBehaviorOptions,\n      );\n    }\n\n    //\n    // All static assets are assumed to have a dot in them\n    //\n    distro.addBehavior(\n      posixPath.join(rootPathPrefix, '/*/*/*.*'),\n      bucketAppsOrigin,\n      s3BehaviorOptions,\n    );\n\n    //\n    // Everything that isn't a static asset is going to API Gateway\n    // There is no trailing slash because Serverless Next.js wants\n    // go load pages at /release/0.0.3 (with no trailing slash).\n    //\n    distro.addBehavior(posixPath.join(rootPathPrefix, '/*'), apiGwyOrigin, apiGwyBehaviorOptions);\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      domainNameOrigin,\n      httpApi,\n      removalPolicy,\n      certEdge,\n      assetNameRoot,\n      assetNameSuffix,\n      r53Zone,\n      bucketLogs,\n      bucketAppsOrigin,\n      rootPathPrefix,\n      createAPIPathRoute = true,\n      createNextDataPathRoute = true,\n      edgeToOriginLambdas,\n    } = props;\n\n    const apigwyOriginRequestPolicy = MicroAppsCF.createAPIOriginPolicy(this, {\n      assetNameRoot,\n      assetNameSuffix,\n      domainNameEdge,\n    });\n\n    //\n    // Determine URL of the origin FQDN\n    //\n    let httpOriginFQDN: string = 'invalid.pwrdrvr.com';\n    if (domainNameOrigin !== undefined) {\n      httpOriginFQDN = domainNameOrigin;\n    } else {\n      httpOriginFQDN = `${httpApi.apiId}.execute-api.${Aws.REGION}.amazonaws.com`;\n    }\n\n    //\n    // Get the Edge to Origin Lambdas\n    //\n\n    //\n    // CloudFront Distro\n    //\n    const apiGwyOrigin = new cforigins.HttpOrigin(httpOriginFQDN, {\n      protocolPolicy: cf.OriginProtocolPolicy.HTTPS_ONLY,\n      originSslProtocols: [cf.OriginSslPolicy.TLS_V1_2],\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: apigwyOriginRequestPolicy,\n        origin: apiGwyOrigin,\n        viewerProtocolPolicy: cf.ViewerProtocolPolicy.REDIRECT_TO_HTTPS,\n        edgeLambdas: edgeToOriginLambdas,\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      apiGwyOrigin,\n      bucketAppsOrigin,\n      distro: this._cloudFrontDistro,\n      apigwyOriginRequestPolicy: apigwyOriginRequestPolicy,\n      rootPathPrefix,\n      createAPIPathRoute,\n      createNextDataPathRoute,\n      apigwyEdgeFunctions: edgeToOriginLambdas,\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"]}
@@ -104,5 +104,13 @@ export declare class MicroAppsEdgeToOrigin extends Construct implements IMicroAp
104
104
  private _edgeToOriginLambdas;
105
105
  get edgeToOriginLambdas(): cf.EdgeLambda[];
106
106
  constructor(scope: Construct, id: string, props: MicroAppsEdgeToOriginProps);
107
+ /**
108
+ * Hash the stack name to make the EdgeFunction parameter name unique
109
+ *
110
+ * @param stack
111
+ * @returns
112
+ */
113
+ private hashStackName;
114
+ private createEdgeFunction;
107
115
  }
108
116
  //# sourceMappingURL=MicroAppsEdgeToOrigin.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"MicroAppsEdgeToOrigin.d.ts","sourceRoot":"","sources":["../src/MicroAppsEdgeToOrigin.ts"],"names":[],"mappings":"AAGA,OAAO,EAAiB,aAAa,EAAE,MAAM,aAAa,CAAC;AAC3D,OAAO,KAAK,EAAE,MAAM,4BAA4B,CAAC;AAEjD,OAAO,KAAK,MAAM,MAAM,wBAAwB,CAAC;AAGjD,OAAO,EAAE,SAAS,EAAE,MAAM,YAAY,CAAC;AAEvC;;GAEG;AACH,MAAM,WAAW,sBAAsB;IACrC;;;;OAIG;IACH,QAAQ,CAAC,oBAAoB,EAAE,MAAM,CAAC,QAAQ,GAAG,EAAE,CAAC,YAAY,CAAC,YAAY,CAAC;IAE9E;;OAEG;IACH,QAAQ,CAAC,mBAAmB,EAAE,EAAE,CAAC,UAAU,EAAE,CAAC;CAC/C;AAED;;GAEG;AACH,MAAM,WAAW,0BAA0B;IACzC;;;;OAIG;IACH,QAAQ,CAAC,aAAa,CAAC,EAAE,aAAa,CAAC;IAEvC;;;;;OAKG;IACH,QAAQ,CAAC,aAAa,CAAC,EAAE,MAAM,CAAC;IAEhC;;;;;OAKG;IACH,QAAQ,CAAC,eAAe,CAAC,EAAE,MAAM,CAAC;IAElC;;;;;;;;OAQG;IACH,QAAQ,CAAC,uBAAuB,CAAC,EAAE,OAAO,CAAC;IAE3C;;;;;;;;;OASG;IACH,QAAQ,CAAC,iBAAiB,CAAC,EAAE,OAAO,CAAC;IAErC;;;;;;;;;;OAUG;IACH,QAAQ,CAAC,WAAW,CAAC,EAAE,MAAM,GAAG,SAAS,GAAG,MAAM,CAAC;IAEnD;;;;;;OAMG;IACH,QAAQ,CAAC,YAAY,CAAC,EAAE,MAAM,CAAC;CAChC;AAED,MAAM,WAAW,iCAAiC;IAChD,QAAQ,CAAC,YAAY,EAAE,MAAM,CAAC;IAC9B,QAAQ,CAAC,WAAW,EAAE,MAAM,GAAG,SAAS,GAAG,EAAE,CAAC;IAC9C,QAAQ,CAAC,uBAAuB,EAAE,OAAO,CAAC;IAC1C,QAAQ,CAAC,iBAAiB,EAAE,OAAO,CAAC;CACrC;AAED;;GAEG;AACH,qBAAa,qBAAsB,SAAQ,SAAU,YAAW,sBAAsB;IACpF;;;;OAIG;WACW,0BAA0B,CAAC,KAAK,EAAE,iCAAiC;IAOjF,OAAO,CAAC,qBAAqB,CAAiD;IAC9E,IAAW,oBAAoB,IAAI,MAAM,CAAC,QAAQ,GAAG,EAAE,CAAC,YAAY,CAAC,YAAY,CAEhF;IAED,OAAO,CAAC,oBAAoB,CAAkB;IAC9C,IAAW,mBAAmB,IAAI,EAAE,CAAC,UAAU,EAAE,CAEhD;gBAEW,KAAK,EAAE,SAAS,EAAE,EAAE,EAAE,MAAM,EAAE,KAAK,EAAE,0BAA0B;CA4I5E"}
1
+ {"version":3,"file":"MicroAppsEdgeToOrigin.d.ts","sourceRoot":"","sources":["../src/MicroAppsEdgeToOrigin.ts"],"names":[],"mappings":"AAIA,OAAO,EAAiB,aAAa,EAAe,MAAM,aAAa,CAAC;AACxE,OAAO,KAAK,EAAE,MAAM,4BAA4B,CAAC;AAEjD,OAAO,KAAK,MAAM,MAAM,wBAAwB,CAAC;AAGjD,OAAO,EAAE,SAAS,EAAE,MAAM,YAAY,CAAC;AAEvC;;GAEG;AACH,MAAM,WAAW,sBAAsB;IACrC;;;;OAIG;IACH,QAAQ,CAAC,oBAAoB,EAAE,MAAM,CAAC,QAAQ,GAAG,EAAE,CAAC,YAAY,CAAC,YAAY,CAAC;IAE9E;;OAEG;IACH,QAAQ,CAAC,mBAAmB,EAAE,EAAE,CAAC,UAAU,EAAE,CAAC;CAC/C;AAED;;GAEG;AACH,MAAM,WAAW,0BAA0B;IACzC;;;;OAIG;IACH,QAAQ,CAAC,aAAa,CAAC,EAAE,aAAa,CAAC;IAEvC;;;;;OAKG;IACH,QAAQ,CAAC,aAAa,CAAC,EAAE,MAAM,CAAC;IAEhC;;;;;OAKG;IACH,QAAQ,CAAC,eAAe,CAAC,EAAE,MAAM,CAAC;IAElC;;;;;;;;OAQG;IACH,QAAQ,CAAC,uBAAuB,CAAC,EAAE,OAAO,CAAC;IAE3C;;;;;;;;;OASG;IACH,QAAQ,CAAC,iBAAiB,CAAC,EAAE,OAAO,CAAC;IAErC;;;;;;;;;;OAUG;IACH,QAAQ,CAAC,WAAW,CAAC,EAAE,MAAM,GAAG,SAAS,GAAG,MAAM,CAAC;IAEnD;;;;;;OAMG;IACH,QAAQ,CAAC,YAAY,CAAC,EAAE,MAAM,CAAC;CAChC;AAED,MAAM,WAAW,iCAAiC;IAChD,QAAQ,CAAC,YAAY,EAAE,MAAM,CAAC;IAC9B,QAAQ,CAAC,WAAW,EAAE,MAAM,GAAG,SAAS,GAAG,EAAE,CAAC;IAC9C,QAAQ,CAAC,uBAAuB,EAAE,OAAO,CAAC;IAC1C,QAAQ,CAAC,iBAAiB,EAAE,OAAO,CAAC;CACrC;AAED;;GAEG;AACH,qBAAa,qBAAsB,SAAQ,SAAU,YAAW,sBAAsB;IACpF;;;;OAIG;WACW,0BAA0B,CAAC,KAAK,EAAE,iCAAiC;IAOjF,OAAO,CAAC,qBAAqB,CAAiD;IAC9E,IAAW,oBAAoB,IAAI,MAAM,CAAC,QAAQ,GAAG,EAAE,CAAC,YAAY,CAAC,YAAY,CAEhF;IAED,OAAO,CAAC,oBAAoB,CAAkB;IAC9C,IAAW,mBAAmB,IAAI,EAAE,CAAC,UAAU,EAAE,CAEhD;gBAEW,KAAK,EAAE,SAAS,EAAE,EAAE,EAAE,MAAM,EAAE,KAAK,EAAE,0BAA0B;IA4I3E;;;;;OAKG;IACH,OAAO,CAAC,aAAa;IAIrB,OAAO,CAAC,kBAAkB;CAwB3B"}
@@ -3,6 +3,7 @@ var _a;
3
3
  Object.defineProperty(exports, "__esModule", { value: true });
4
4
  exports.MicroAppsEdgeToOrigin = void 0;
5
5
  const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
6
+ const crypto = require("crypto");
6
7
  const fs_1 = require("fs");
7
8
  const os = require("os");
8
9
  const path = require("path");
@@ -37,7 +38,7 @@ class MicroAppsEdgeToOrigin extends constructs_1.Construct {
37
38
  functionName: assetNameRoot ? `${assetNameRoot}-edge-to-origin${assetNameSuffix}` : undefined,
38
39
  memorySize: 1769,
39
40
  logRetention: logs.RetentionDays.ONE_MONTH,
40
- runtime: lambda.Runtime.NODEJS_14_X,
41
+ runtime: lambda.Runtime.NODEJS_16_X,
41
42
  timeout: aws_cdk_lib_1.Duration.seconds(5),
42
43
  initialPolicy: [
43
44
  // This can't have a reference to the httpApi because it would mean
@@ -54,40 +55,36 @@ class MicroAppsEdgeToOrigin extends constructs_1.Construct {
54
55
  ],
55
56
  ...(removalPolicy ? { removalPolicy } : {}),
56
57
  };
57
- if (process.env.NODE_ENV === 'test' &&
58
- fs_1.existsSync(path.join(__dirname, '..', '..', 'microapps-edge-to-origin', 'dist', 'index.js'))) {
59
- // Emit the config file from the construct options
60
- fs_1.writeFileSync(path.join(__dirname, '..', '..', 'microapps-edge-to-origin', 'dist', 'config.yml'), edgeToOriginConfigYaml);
61
- // copyFileSync(
62
- // path.join(__dirname, '..', '..', '..', 'configs', 'microapps-edge-to-origin', 'config.yml'),
63
- // path.join(__dirname, '..', '..', 'microapps-edge-to-origin', 'dist', 'config.yml'),
64
- // );
65
- // This is for tests run under jest
58
+ const rootDistPath = path.join(__dirname, '..', '..', 'microapps-edge-to-origin', 'dist');
59
+ const rootDistExists = fs_1.existsSync(path.join(rootDistPath, 'index.js'));
60
+ const localDistPath = path.join(__dirname, 'microapps-edge-to-origin');
61
+ const localDistExists = fs_1.existsSync(path.join(localDistPath, 'index.js'));
62
+ if (process.env.NODE_ENV === 'test' && rootDistExists) {
63
+ // This is for tests run under jest - Prefer root dist bundle
66
64
  // This is also for anytime when the edge function has already been bundled
67
- this._edgeToOriginFunction = new cf.experimental.EdgeFunction(this, 'edge-to-apigwy-func', {
68
- code: lambda.Code.fromAsset(path.join(__dirname, '..', '..', 'microapps-edge-to-origin', 'dist')),
69
- handler: 'index.handler',
70
- ...edgeToOriginFuncProps,
71
- });
65
+ this._edgeToOriginFunction = this.createEdgeFunction(rootDistPath, edgeToOriginConfigYaml, edgeToOriginFuncProps);
72
66
  }
73
- else if (fs_1.existsSync(path.join(__dirname, 'microapps-edge-to-origin', 'index.js'))) {
74
- // Emit the config file from the construct options
75
- fs_1.writeFileSync(path.join(__dirname, 'microapps-edge-to-origin', 'config.yml'), edgeToOriginConfigYaml);
76
- // This is for built apps packaged with the CDK construct
77
- this._edgeToOriginFunction = new cf.experimental.EdgeFunction(this, 'edge-to-apigwy-func', {
78
- code: lambda.Code.fromAsset(path.join(__dirname, 'microapps-edge-to-origin')),
79
- handler: 'index.handler',
80
- ...edgeToOriginFuncProps,
81
- });
67
+ else if (localDistExists) {
68
+ // Prefer local dist above root dist if both exist (when building for distribution)
69
+ this._edgeToOriginFunction = this.createEdgeFunction(localDistPath, edgeToOriginConfigYaml, edgeToOriginFuncProps);
70
+ }
71
+ else if (rootDistExists) {
72
+ // Use local dist if it exists (when deploying from CDK in this repo)
73
+ this._edgeToOriginFunction = this.createEdgeFunction(rootDistPath, edgeToOriginConfigYaml, edgeToOriginFuncProps);
82
74
  }
83
75
  else {
76
+ // 2022-07-30 - Does this actually get used at all anymore?
77
+ // 2022-10-02 - This is broken - it's emitting a config file but then
78
+ // usinga different config file in the bundling below.
79
+ // This may be ok if this is only used for the construct packaging
80
+ // as the consuming stack should select a different above which will
81
+ // use the correct config file.
84
82
  // Emit the config file from the construct options
85
83
  fs_1.writeFileSync(path.join(__dirname, '..', '..', 'microapps-edge-to-origin', 'config.yml'), edgeToOriginConfigYaml);
86
84
  // This builds the function for distribution with the CDK Construct
87
85
  // and will be used during local builds and PR builds of microapps-core
88
86
  // if the microapps-edge-to-origin function is not already bundled.
89
87
  // This will fail to deploy in any region other than us-east-1
90
- // We cannot use NodejsFunction because it will not create in us-east-1
91
88
  this._edgeToOriginFunction = new lambdaNodejs.NodejsFunction(this, 'edge-to-apigwy-func', {
92
89
  entry: path.join(__dirname, '..', '..', 'microapps-edge-to-origin', 'src', 'index.ts'),
93
90
  handler: 'handler',
@@ -98,6 +95,8 @@ class MicroAppsEdgeToOrigin extends constructs_1.Construct {
98
95
  beforeInstall: () => [],
99
96
  beforeBundling: () => [],
100
97
  afterBundling: (_inputDir, outputDir) => {
98
+ // 2022-10-02 - Note that this is ignoring the generated config
99
+ // file above and including the default template config file
101
100
  return [
102
101
  `${os.platform() === 'win32' ? 'copy' : 'cp'} ${path.join(__dirname, '..', '..', '..', 'configs', 'microapps-edge-to-origin', 'config.yml')} ${outputDir}`,
103
102
  ];
@@ -132,8 +131,31 @@ replaceHostHeader: ${props.replaceHostHeader}`;
132
131
  get edgeToOriginLambdas() {
133
132
  return this._edgeToOriginLambdas;
134
133
  }
134
+ /**
135
+ * Hash the stack name to make the EdgeFunction parameter name unique
136
+ *
137
+ * @param stack
138
+ * @returns
139
+ */
140
+ hashStackName() {
141
+ return crypto.createHash('sha1').update(aws_cdk_lib_1.Stack.of(this).stackName).digest('hex').substring(0, 8);
142
+ }
143
+ createEdgeFunction(distPath, edgeToOriginConfigYaml, edgeToOriginFuncProps) {
144
+ fs_1.writeFileSync(path.join(distPath, 'config.yml'), edgeToOriginConfigYaml);
145
+ // EdgeFunction has a bug where it will generate the same parameter
146
+ // name across multiple stacks in the same region if the id param is constant
147
+ const edge = new cf.experimental.EdgeFunction(this, `edge-to-apigwy-func-${this.hashStackName()}`, {
148
+ stackId: `microapps-edge-to-origin-${this.hashStackName()}`,
149
+ code: lambda.Code.fromAsset(distPath),
150
+ functionName: `microapps-edge-to-origin-${this.hashStackName()}`,
151
+ handler: 'index.handler',
152
+ ...edgeToOriginFuncProps,
153
+ });
154
+ aws_cdk_lib_1.Tags.of(edge).add('Name', aws_cdk_lib_1.Stack.of(this).stackName);
155
+ return edge;
156
+ }
135
157
  }
136
158
  exports.MicroAppsEdgeToOrigin = MicroAppsEdgeToOrigin;
137
159
  _a = JSII_RTTI_SYMBOL_1;
138
- MicroAppsEdgeToOrigin[_a] = { fqn: "@pwrdrvr/microapps-cdk.MicroAppsEdgeToOrigin", version: "0.2.12" };
139
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"MicroAppsEdgeToOrigin.js","sourceRoot":"","sources":["../src/MicroAppsEdgeToOrigin.ts"],"names":[],"mappings":"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","sourcesContent":["import { existsSync, writeFileSync } from 'fs';\nimport * as os from 'os';\nimport * as path from 'path';\nimport { Aws, Duration, RemovalPolicy } from 'aws-cdk-lib';\nimport * as cf from 'aws-cdk-lib/aws-cloudfront';\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 * Represents a MicroApps Edge to Origin Function\n */\nexport interface IMicroAppsEdgeToOrigin {\n  /**\n   * The edge to origin function for API Gateway Request Origin Edge Lambda\n   *\n   * The generated `config.yml` is included in the Lambda's code.\n   */\n  readonly edgeToOriginFunction: lambda.Function | cf.experimental.EdgeFunction;\n\n  /**\n   * Configuration of the edge to origin lambda functions\n   */\n  readonly edgeToOriginLambdas: cf.EdgeLambda[];\n}\n\n/**\n * Properties to initialize an instance of `MicroAppsEdgeToOrigin`.\n */\nexport interface MicroAppsEdgeToOriginProps {\n  /**\n   * RemovalPolicy override for child resources\n   *\n   * @default - per resource default\n   */\n  readonly removalPolicy?: RemovalPolicy;\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   * 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 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\nexport interface GenerateEdgeToOriginConfigOptions {\n  readonly originRegion: string;\n  readonly signingMode: 'sign' | 'presign' | '';\n  readonly addXForwardedHostHeader: boolean;\n  readonly replaceHostHeader: boolean;\n}\n\n/**\n * Create a new MicroApps Edge to Origin Function w/ `config.yml`\n */\nexport class MicroAppsEdgeToOrigin extends Construct implements IMicroAppsEdgeToOrigin {\n  /**\n   * Generate the yaml config for the edge lambda\n   * @param props\n   * @returns\n   */\n  public static generateEdgeToOriginConfig(props: GenerateEdgeToOriginConfigOptions) {\n    return `originRegion: ${props.originRegion}\n${props.signingMode === '' ? '' : `signingMode: ${props.signingMode}`}\naddXForwardedHostHeader: ${props.addXForwardedHostHeader}\nreplaceHostHeader: ${props.replaceHostHeader}`;\n  }\n\n  private _edgeToOriginFunction: lambda.Function | cf.experimental.EdgeFunction;\n  public get edgeToOriginFunction(): lambda.Function | cf.experimental.EdgeFunction {\n    return this._edgeToOriginFunction;\n  }\n\n  private _edgeToOriginLambdas: cf.EdgeLambda[];\n  public get edgeToOriginLambdas(): cf.EdgeLambda[] {\n    return this._edgeToOriginLambdas;\n  }\n\n  constructor(scope: Construct, id: string, props: MicroAppsEdgeToOriginProps) {\n    super(scope, id);\n\n    if (props === undefined) {\n      throw new Error('props must be set');\n    }\n\n    const {\n      removalPolicy,\n      assetNameRoot,\n      assetNameSuffix,\n      signingMode = 'sign',\n      addXForwardedHostHeader = true,\n      replaceHostHeader = true,\n      originRegion,\n    } = props;\n\n    // Create the edge function config file from the construct options\n    const edgeToOriginConfigYaml = MicroAppsEdgeToOrigin.generateEdgeToOriginConfig({\n      originRegion: originRegion || Aws.REGION,\n      addXForwardedHostHeader,\n      replaceHostHeader,\n      signingMode: signingMode === 'none' ? '' : signingMode,\n    });\n\n    //\n    // Create the Edge to Origin Function\n    //\n    const edgeToOriginFuncProps: Omit<lambda.FunctionProps, 'handler' | 'code'> = {\n      functionName: assetNameRoot ? `${assetNameRoot}-edge-to-origin${assetNameSuffix}` : undefined,\n      memorySize: 1769,\n      logRetention: logs.RetentionDays.ONE_MONTH,\n      runtime: lambda.Runtime.NODEJS_14_X,\n      timeout: Duration.seconds(5),\n      initialPolicy: [\n        // This can't have a reference to the httpApi because it would mean\n        // the parent stack (this stack) has to be created before the us-east-1\n        // child stack for the Edge Lambda Function.\n        // That's why we use a tag-based policy to allow the Edge Function\n        // to invoke any API Gateway API that we apply a tag to\n        // We allow the edge function to sign for all regions since\n        // we may use custom closest region in the future.\n        new iam.PolicyStatement({\n          actions: ['execute-api:Invoke'],\n          resources: [`arn:aws:execute-api:*:${Aws.ACCOUNT_ID}:*/*/*/*`],\n          // Unfortunately, API Gateway access cannot be restricted using\n          // tags on the target resource\n          // https://docs.aws.amazon.com/IAM/latest/UserGuide/access_tags.html\n          // https://docs.aws.amazon.com/IAM/latest/UserGuide/reference_aws-services-that-work-with-iam.html#networking_svcs\n          // conditions: {\n          //   // TODO: Set this to a string unique to each stack\n          //   StringEquals: { 'aws:ResourceTag/microapp-managed': 'true' },\n          // },\n        }),\n      ],\n      ...(removalPolicy ? { removalPolicy } : {}),\n    };\n    if (\n      process.env.NODE_ENV === 'test' &&\n      existsSync(path.join(__dirname, '..', '..', 'microapps-edge-to-origin', 'dist', 'index.js'))\n    ) {\n      // Emit the config file from the construct options\n      writeFileSync(\n        path.join(__dirname, '..', '..', 'microapps-edge-to-origin', 'dist', 'config.yml'),\n        edgeToOriginConfigYaml,\n      );\n      // copyFileSync(\n      //   path.join(__dirname, '..', '..', '..', 'configs', 'microapps-edge-to-origin', 'config.yml'),\n      //   path.join(__dirname, '..', '..', 'microapps-edge-to-origin', 'dist', 'config.yml'),\n      // );\n      // This is for tests run under jest\n      // This is also for anytime when the edge function has already been bundled\n      this._edgeToOriginFunction = new cf.experimental.EdgeFunction(this, 'edge-to-apigwy-func', {\n        code: lambda.Code.fromAsset(\n          path.join(__dirname, '..', '..', 'microapps-edge-to-origin', 'dist'),\n        ),\n        handler: 'index.handler',\n        ...edgeToOriginFuncProps,\n      });\n    } else if (existsSync(path.join(__dirname, 'microapps-edge-to-origin', 'index.js'))) {\n      // Emit the config file from the construct options\n      writeFileSync(\n        path.join(__dirname, 'microapps-edge-to-origin', 'config.yml'),\n        edgeToOriginConfigYaml,\n      );\n\n      // This is for built apps packaged with the CDK construct\n      this._edgeToOriginFunction = new cf.experimental.EdgeFunction(this, 'edge-to-apigwy-func', {\n        code: lambda.Code.fromAsset(path.join(__dirname, 'microapps-edge-to-origin')),\n        handler: 'index.handler',\n        ...edgeToOriginFuncProps,\n      });\n    } else {\n      // Emit the config file from the construct options\n      writeFileSync(\n        path.join(__dirname, '..', '..', 'microapps-edge-to-origin', 'config.yml'),\n        edgeToOriginConfigYaml,\n      );\n\n      // This builds the function for distribution with the CDK Construct\n      // and will be used during local builds and PR builds of microapps-core\n      // if the microapps-edge-to-origin function is not already bundled.\n      // This will fail to deploy in any region other than us-east-1\n      // We cannot use NodejsFunction because it will not create in us-east-1\n      this._edgeToOriginFunction = new lambdaNodejs.NodejsFunction(this, 'edge-to-apigwy-func', {\n        entry: path.join(__dirname, '..', '..', 'microapps-edge-to-origin', 'src', 'index.ts'),\n        handler: 'handler',\n        bundling: {\n          minify: true,\n          sourceMap: true,\n          commandHooks: {\n            beforeInstall: () => [],\n            beforeBundling: () => [],\n            afterBundling: (_inputDir: string, outputDir: string) => {\n              return [\n                `${os.platform() === 'win32' ? 'copy' : 'cp'} ${path.join(\n                  __dirname,\n                  '..',\n                  '..',\n                  '..',\n                  'configs',\n                  'microapps-edge-to-origin',\n                  'config.yml',\n                )} ${outputDir}`,\n              ];\n            },\n          },\n        },\n        ...edgeToOriginFuncProps,\n      });\n    }\n\n    this._edgeToOriginLambdas = [\n      {\n        eventType: cf.LambdaEdgeEventType.ORIGIN_REQUEST,\n        functionVersion: this._edgeToOriginFunction.currentVersion,\n        includeBody: true,\n      },\n    ];\n  }\n}\n"]}
160
+ MicroAppsEdgeToOrigin[_a] = { fqn: "@pwrdrvr/microapps-cdk.MicroAppsEdgeToOrigin", version: "0.2.13" };
161
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"MicroAppsEdgeToOrigin.js","sourceRoot":"","sources":["../src/MicroAppsEdgeToOrigin.ts"],"names":[],"mappings":";;;;;AAAA,iCAAiC;AACjC,2BAA+C;AAC/C,yBAAyB;AACzB,6BAA6B;AAC7B,6CAAwE;AACxE,iDAAiD;AACjD,2CAA2C;AAC3C,iDAAiD;AACjD,8DAA8D;AAC9D,6CAA6C;AAC7C,2CAAuC;AAmGvC;;GAEG;AACH,MAAa,qBAAsB,SAAQ,sBAAS;IAuBlD,YAAY,KAAgB,EAAE,EAAU,EAAE,KAAiC;QACzE,KAAK,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;QAEjB,IAAI,KAAK,KAAK,SAAS,EAAE;YACvB,MAAM,IAAI,KAAK,CAAC,mBAAmB,CAAC,CAAC;SACtC;QAED,MAAM,EACJ,aAAa,EACb,aAAa,EACb,eAAe,EACf,WAAW,GAAG,MAAM,EACpB,uBAAuB,GAAG,IAAI,EAC9B,iBAAiB,GAAG,IAAI,EACxB,YAAY,GACb,GAAG,KAAK,CAAC;QAEV,kEAAkE;QAClE,MAAM,sBAAsB,GAAG,qBAAqB,CAAC,0BAA0B,CAAC;YAC9E,YAAY,EAAE,YAAY,IAAI,iBAAG,CAAC,MAAM;YACxC,uBAAuB;YACvB,iBAAiB;YACjB,WAAW,EAAE,WAAW,KAAK,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,WAAW;SACvD,CAAC,CAAC;QAEH,EAAE;QACF,qCAAqC;QACrC,EAAE;QACF,MAAM,qBAAqB,GAAmD;YAC5E,YAAY,EAAE,aAAa,CAAC,CAAC,CAAC,GAAG,aAAa,kBAAkB,eAAe,EAAE,CAAC,CAAC,CAAC,SAAS;YAC7F,UAAU,EAAE,IAAI;YAChB,YAAY,EAAE,IAAI,CAAC,aAAa,CAAC,SAAS;YAC1C,OAAO,EAAE,MAAM,CAAC,OAAO,CAAC,WAAW;YACnC,OAAO,EAAE,sBAAQ,CAAC,OAAO,CAAC,CAAC,CAAC;YAC5B,aAAa,EAAE;gBACb,mEAAmE;gBACnE,uEAAuE;gBACvE,4CAA4C;gBAC5C,kEAAkE;gBAClE,uDAAuD;gBACvD,2DAA2D;gBAC3D,kDAAkD;gBAClD,IAAI,GAAG,CAAC,eAAe,CAAC;oBACtB,OAAO,EAAE,CAAC,oBAAoB,CAAC;oBAC/B,SAAS,EAAE,CAAC,yBAAyB,iBAAG,CAAC,UAAU,UAAU,CAAC;iBAS/D,CAAC;aACH;YACD,GAAG,CAAC,aAAa,CAAC,CAAC,CAAC,EAAE,aAAa,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC;SAC5C,CAAC;QACF,MAAM,YAAY,GAAG,IAAI,CAAC,IAAI,CAAC,SAAS,EAAE,IAAI,EAAE,IAAI,EAAE,0BAA0B,EAAE,MAAM,CAAC,CAAC;QAC1F,MAAM,cAAc,GAAG,eAAU,CAAC,IAAI,CAAC,IAAI,CAAC,YAAY,EAAE,UAAU,CAAC,CAAC,CAAC;QACvE,MAAM,aAAa,GAAG,IAAI,CAAC,IAAI,CAAC,SAAS,EAAE,0BAA0B,CAAC,CAAC;QACvE,MAAM,eAAe,GAAG,eAAU,CAAC,IAAI,CAAC,IAAI,CAAC,aAAa,EAAE,UAAU,CAAC,CAAC,CAAC;QACzE,IAAI,OAAO,CAAC,GAAG,CAAC,QAAQ,KAAK,MAAM,IAAI,cAAc,EAAE;YACrD,6DAA6D;YAC7D,2EAA2E;YAC3E,IAAI,CAAC,qBAAqB,GAAG,IAAI,CAAC,kBAAkB,CAClD,YAAY,EACZ,sBAAsB,EACtB,qBAAqB,CACtB,CAAC;SACH;aAAM,IAAI,eAAe,EAAE;YAC1B,mFAAmF;YACnF,IAAI,CAAC,qBAAqB,GAAG,IAAI,CAAC,kBAAkB,CAClD,aAAa,EACb,sBAAsB,EACtB,qBAAqB,CACtB,CAAC;SACH;aAAM,IAAI,cAAc,EAAE;YACzB,qEAAqE;YACrE,IAAI,CAAC,qBAAqB,GAAG,IAAI,CAAC,kBAAkB,CAClD,YAAY,EACZ,sBAAsB,EACtB,qBAAqB,CACtB,CAAC;SACH;aAAM;YACL,2DAA2D;YAE3D,qEAAqE;YACrE,sDAAsD;YACtD,kEAAkE;YAClE,oEAAoE;YACpE,+BAA+B;YAC/B,kDAAkD;YAClD,kBAAa,CACX,IAAI,CAAC,IAAI,CAAC,SAAS,EAAE,IAAI,EAAE,IAAI,EAAE,0BAA0B,EAAE,YAAY,CAAC,EAC1E,sBAAsB,CACvB,CAAC;YAEF,mEAAmE;YACnE,uEAAuE;YACvE,mEAAmE;YACnE,8DAA8D;YAC9D,IAAI,CAAC,qBAAqB,GAAG,IAAI,YAAY,CAAC,cAAc,CAAC,IAAI,EAAE,qBAAqB,EAAE;gBACxF,KAAK,EAAE,IAAI,CAAC,IAAI,CAAC,SAAS,EAAE,IAAI,EAAE,IAAI,EAAE,0BAA0B,EAAE,KAAK,EAAE,UAAU,CAAC;gBACtF,OAAO,EAAE,SAAS;gBAClB,QAAQ,EAAE;oBACR,MAAM,EAAE,IAAI;oBACZ,SAAS,EAAE,IAAI;oBACf,YAAY,EAAE;wBACZ,aAAa,EAAE,GAAG,EAAE,CAAC,EAAE;wBACvB,cAAc,EAAE,GAAG,EAAE,CAAC,EAAE;wBACxB,aAAa,EAAE,CAAC,SAAiB,EAAE,SAAiB,EAAE,EAAE;4BACtD,+DAA+D;4BAC/D,4DAA4D;4BAC5D,OAAO;gCACL,GAAG,EAAE,CAAC,QAAQ,EAAE,KAAK,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,IAAI,IAAI,IAAI,CAAC,IAAI,CACvD,SAAS,EACT,IAAI,EACJ,IAAI,EACJ,IAAI,EACJ,SAAS,EACT,0BAA0B,EAC1B,YAAY,CACb,IAAI,SAAS,EAAE;6BACjB,CAAC;wBACJ,CAAC;qBACF;iBACF;gBACD,GAAG,qBAAqB;aACzB,CAAC,CAAC;SACJ;QAED,IAAI,CAAC,oBAAoB,GAAG;YAC1B;gBACE,SAAS,EAAE,EAAE,CAAC,mBAAmB,CAAC,cAAc;gBAChD,eAAe,EAAE,IAAI,CAAC,qBAAqB,CAAC,cAAc;gBAC1D,WAAW,EAAE,IAAI;aAClB;SACF,CAAC;IACJ,CAAC;IAhKD;;;;OAIG;IACI,MAAM,CAAC,0BAA0B,CAAC,KAAwC;QAC/E,OAAO,iBAAiB,KAAK,CAAC,YAAY;EAC5C,KAAK,CAAC,WAAW,KAAK,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,gBAAgB,KAAK,CAAC,WAAW,EAAE;2BAC1C,KAAK,CAAC,uBAAuB;qBACnC,KAAK,CAAC,iBAAiB,EAAE,CAAC;IAC7C,CAAC;IAGD,IAAW,oBAAoB;QAC7B,OAAO,IAAI,CAAC,qBAAqB,CAAC;IACpC,CAAC;IAGD,IAAW,mBAAmB;QAC5B,OAAO,IAAI,CAAC,oBAAoB,CAAC;IACnC,CAAC;IA8ID;;;;;OAKG;IACK,aAAa;QACnB,OAAO,MAAM,CAAC,UAAU,CAAC,MAAM,CAAC,CAAC,MAAM,CAAC,mBAAK,CAAC,EAAE,CAAC,IAAI,CAAC,CAAC,SAAS,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,SAAS,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC;IAClG,CAAC;IAEO,kBAAkB,CACxB,QAAgB,EAChB,sBAA8B,EAC9B,qBAAqE;QAErE,kBAAa,CAAC,IAAI,CAAC,IAAI,CAAC,QAAQ,EAAE,YAAY,CAAC,EAAE,sBAAsB,CAAC,CAAC;QAEzE,mEAAmE;QACnE,6EAA6E;QAC7E,MAAM,IAAI,GAAG,IAAI,EAAE,CAAC,YAAY,CAAC,YAAY,CAC3C,IAAI,EACJ,uBAAuB,IAAI,CAAC,aAAa,EAAE,EAAE,EAC7C;YACE,OAAO,EAAE,4BAA4B,IAAI,CAAC,aAAa,EAAE,EAAE;YAC3D,IAAI,EAAE,MAAM,CAAC,IAAI,CAAC,SAAS,CAAC,QAAQ,CAAC;YACrC,YAAY,EAAE,4BAA4B,IAAI,CAAC,aAAa,EAAE,EAAE;YAChE,OAAO,EAAE,eAAe;YACxB,GAAG,qBAAqB;SACzB,CACF,CAAC;QACF,kBAAI,CAAC,EAAE,CAAC,IAAI,CAAC,CAAC,GAAG,CAAC,MAAM,EAAE,mBAAK,CAAC,EAAE,CAAC,IAAI,CAAC,CAAC,SAAS,CAAC,CAAC;QAEpD,OAAO,IAAI,CAAC;IACd,CAAC;;AApMH,sDAqMC","sourcesContent":["import * as crypto from 'crypto';\nimport { existsSync, writeFileSync } from 'fs';\nimport * as os from 'os';\nimport * as path from 'path';\nimport { Aws, Duration, RemovalPolicy, Stack, Tags } from 'aws-cdk-lib';\nimport * as cf from 'aws-cdk-lib/aws-cloudfront';\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 * Represents a MicroApps Edge to Origin Function\n */\nexport interface IMicroAppsEdgeToOrigin {\n  /**\n   * The edge to origin function for API Gateway Request Origin Edge Lambda\n   *\n   * The generated `config.yml` is included in the Lambda's code.\n   */\n  readonly edgeToOriginFunction: lambda.Function | cf.experimental.EdgeFunction;\n\n  /**\n   * Configuration of the edge to origin lambda functions\n   */\n  readonly edgeToOriginLambdas: cf.EdgeLambda[];\n}\n\n/**\n * Properties to initialize an instance of `MicroAppsEdgeToOrigin`.\n */\nexport interface MicroAppsEdgeToOriginProps {\n  /**\n   * RemovalPolicy override for child resources\n   *\n   * @default - per resource default\n   */\n  readonly removalPolicy?: RemovalPolicy;\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   * 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 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\nexport interface GenerateEdgeToOriginConfigOptions {\n  readonly originRegion: string;\n  readonly signingMode: 'sign' | 'presign' | '';\n  readonly addXForwardedHostHeader: boolean;\n  readonly replaceHostHeader: boolean;\n}\n\n/**\n * Create a new MicroApps Edge to Origin Function w/ `config.yml`\n */\nexport class MicroAppsEdgeToOrigin extends Construct implements IMicroAppsEdgeToOrigin {\n  /**\n   * Generate the yaml config for the edge lambda\n   * @param props\n   * @returns\n   */\n  public static generateEdgeToOriginConfig(props: GenerateEdgeToOriginConfigOptions) {\n    return `originRegion: ${props.originRegion}\n${props.signingMode === '' ? '' : `signingMode: ${props.signingMode}`}\naddXForwardedHostHeader: ${props.addXForwardedHostHeader}\nreplaceHostHeader: ${props.replaceHostHeader}`;\n  }\n\n  private _edgeToOriginFunction: lambda.Function | cf.experimental.EdgeFunction;\n  public get edgeToOriginFunction(): lambda.Function | cf.experimental.EdgeFunction {\n    return this._edgeToOriginFunction;\n  }\n\n  private _edgeToOriginLambdas: cf.EdgeLambda[];\n  public get edgeToOriginLambdas(): cf.EdgeLambda[] {\n    return this._edgeToOriginLambdas;\n  }\n\n  constructor(scope: Construct, id: string, props: MicroAppsEdgeToOriginProps) {\n    super(scope, id);\n\n    if (props === undefined) {\n      throw new Error('props must be set');\n    }\n\n    const {\n      removalPolicy,\n      assetNameRoot,\n      assetNameSuffix,\n      signingMode = 'sign',\n      addXForwardedHostHeader = true,\n      replaceHostHeader = true,\n      originRegion,\n    } = props;\n\n    // Create the edge function config file from the construct options\n    const edgeToOriginConfigYaml = MicroAppsEdgeToOrigin.generateEdgeToOriginConfig({\n      originRegion: originRegion || Aws.REGION,\n      addXForwardedHostHeader,\n      replaceHostHeader,\n      signingMode: signingMode === 'none' ? '' : signingMode,\n    });\n\n    //\n    // Create the Edge to Origin Function\n    //\n    const edgeToOriginFuncProps: Omit<lambda.FunctionProps, 'handler' | 'code'> = {\n      functionName: assetNameRoot ? `${assetNameRoot}-edge-to-origin${assetNameSuffix}` : undefined,\n      memorySize: 1769,\n      logRetention: logs.RetentionDays.ONE_MONTH,\n      runtime: lambda.Runtime.NODEJS_16_X,\n      timeout: Duration.seconds(5),\n      initialPolicy: [\n        // This can't have a reference to the httpApi because it would mean\n        // the parent stack (this stack) has to be created before the us-east-1\n        // child stack for the Edge Lambda Function.\n        // That's why we use a tag-based policy to allow the Edge Function\n        // to invoke any API Gateway API that we apply a tag to\n        // We allow the edge function to sign for all regions since\n        // we may use custom closest region in the future.\n        new iam.PolicyStatement({\n          actions: ['execute-api:Invoke'],\n          resources: [`arn:aws:execute-api:*:${Aws.ACCOUNT_ID}:*/*/*/*`],\n          // Unfortunately, API Gateway access cannot be restricted using\n          // tags on the target resource\n          // https://docs.aws.amazon.com/IAM/latest/UserGuide/access_tags.html\n          // https://docs.aws.amazon.com/IAM/latest/UserGuide/reference_aws-services-that-work-with-iam.html#networking_svcs\n          // conditions: {\n          //   // TODO: Set this to a string unique to each stack\n          //   StringEquals: { 'aws:ResourceTag/microapp-managed': 'true' },\n          // },\n        }),\n      ],\n      ...(removalPolicy ? { removalPolicy } : {}),\n    };\n    const rootDistPath = path.join(__dirname, '..', '..', 'microapps-edge-to-origin', 'dist');\n    const rootDistExists = existsSync(path.join(rootDistPath, 'index.js'));\n    const localDistPath = path.join(__dirname, 'microapps-edge-to-origin');\n    const localDistExists = existsSync(path.join(localDistPath, 'index.js'));\n    if (process.env.NODE_ENV === 'test' && rootDistExists) {\n      // This is for tests run under jest - Prefer root dist bundle\n      // This is also for anytime when the edge function has already been bundled\n      this._edgeToOriginFunction = this.createEdgeFunction(\n        rootDistPath,\n        edgeToOriginConfigYaml,\n        edgeToOriginFuncProps,\n      );\n    } else if (localDistExists) {\n      // Prefer local dist above root dist if both exist (when building for distribution)\n      this._edgeToOriginFunction = this.createEdgeFunction(\n        localDistPath,\n        edgeToOriginConfigYaml,\n        edgeToOriginFuncProps,\n      );\n    } else if (rootDistExists) {\n      // Use local dist if it exists (when deploying from CDK in this repo)\n      this._edgeToOriginFunction = this.createEdgeFunction(\n        rootDistPath,\n        edgeToOriginConfigYaml,\n        edgeToOriginFuncProps,\n      );\n    } else {\n      // 2022-07-30 - Does this actually get used at all anymore?\n\n      // 2022-10-02 - This is broken - it's emitting a config file but then\n      // usinga different config file in the bundling below.\n      // This may be ok if this is only used for the construct packaging\n      // as the consuming stack should select a different above which will\n      // use the correct config file.\n      // Emit the config file from the construct options\n      writeFileSync(\n        path.join(__dirname, '..', '..', 'microapps-edge-to-origin', 'config.yml'),\n        edgeToOriginConfigYaml,\n      );\n\n      // This builds the function for distribution with the CDK Construct\n      // and will be used during local builds and PR builds of microapps-core\n      // if the microapps-edge-to-origin function is not already bundled.\n      // This will fail to deploy in any region other than us-east-1\n      this._edgeToOriginFunction = new lambdaNodejs.NodejsFunction(this, 'edge-to-apigwy-func', {\n        entry: path.join(__dirname, '..', '..', 'microapps-edge-to-origin', 'src', 'index.ts'),\n        handler: 'handler',\n        bundling: {\n          minify: true,\n          sourceMap: true,\n          commandHooks: {\n            beforeInstall: () => [],\n            beforeBundling: () => [],\n            afterBundling: (_inputDir: string, outputDir: string) => {\n              // 2022-10-02 - Note that this is ignoring the generated config\n              // file above and including the default template config file\n              return [\n                `${os.platform() === 'win32' ? 'copy' : 'cp'} ${path.join(\n                  __dirname,\n                  '..',\n                  '..',\n                  '..',\n                  'configs',\n                  'microapps-edge-to-origin',\n                  'config.yml',\n                )} ${outputDir}`,\n              ];\n            },\n          },\n        },\n        ...edgeToOriginFuncProps,\n      });\n    }\n\n    this._edgeToOriginLambdas = [\n      {\n        eventType: cf.LambdaEdgeEventType.ORIGIN_REQUEST,\n        functionVersion: this._edgeToOriginFunction.currentVersion,\n        includeBody: true,\n      },\n    ];\n  }\n\n  /**\n   * Hash the stack name to make the EdgeFunction parameter name unique\n   *\n   * @param stack\n   * @returns\n   */\n  private hashStackName(): string {\n    return crypto.createHash('sha1').update(Stack.of(this).stackName).digest('hex').substring(0, 8);\n  }\n\n  private createEdgeFunction(\n    distPath: string,\n    edgeToOriginConfigYaml: string,\n    edgeToOriginFuncProps: Omit<lambda.FunctionProps, 'handler' | 'code'>,\n  ) {\n    writeFileSync(path.join(distPath, 'config.yml'), edgeToOriginConfigYaml);\n\n    // EdgeFunction has a bug where it will generate the same parameter\n    // name across multiple stacks in the same region if the id param is constant\n    const edge = new cf.experimental.EdgeFunction(\n      this,\n      `edge-to-apigwy-func-${this.hashStackName()}`,\n      {\n        stackId: `microapps-edge-to-origin-${this.hashStackName()}`,\n        code: lambda.Code.fromAsset(distPath),\n        functionName: `microapps-edge-to-origin-${this.hashStackName()}`,\n        handler: 'index.handler',\n        ...edgeToOriginFuncProps,\n      },\n    );\n    Tags.of(edge).add('Name', Stack.of(this).stackName);\n\n    return edge;\n  }\n}\n"]}
@@ -74,5 +74,5 @@ class MicroAppsS3 extends constructs_1.Construct {
74
74
  }
75
75
  exports.MicroAppsS3 = MicroAppsS3;
76
76
  _a = JSII_RTTI_SYMBOL_1;
77
- MicroAppsS3[_a] = { fqn: "@pwrdrvr/microapps-cdk.MicroAppsS3", version: "0.2.12" };
77
+ MicroAppsS3[_a] = { fqn: "@pwrdrvr/microapps-cdk.MicroAppsS3", version: "0.2.13" };
78
78
  //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"MicroAppsS3.js","sourceRoot":"","sources":["../src/MicroAppsS3.ts"],"names":[],"mappings":";;;;;AAAA,6CAA4C;AAC5C,iDAAiD;AACjD,gEAAgE;AAChE,yCAAyC;AACzC,2CAAuC;AAmFvC;;;;;GAKG;AACH,MAAa,WAAY,SAAQ,sBAAS;IA0BxC,YAAY,KAAgB,EAAE,EAAU,EAAE,KAAwB;QAChE,KAAK,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;QAEjB,IAAI,KAAK,KAAK,SAAS,EAAE;YACvB,MAAM,IAAI,KAAK,CAAC,mBAAmB,CAAC,CAAC;SACtC;QAED,MAAM,EAAE,aAAa,EAAE,aAAa,EAAE,eAAe,EAAE,GAAG,KAAK,CAAC;QAEhE,wDAAwD;QACxD,MAAM,iBAAiB,GAAG,aAAa,KAAK,2BAAa,CAAC,OAAO,CAAC;QAElE,EAAE;QACF,gDAAgD;QAChD,EAAE;QACF,IAAI,CAAC,WAAW,GAAG,IAAI,EAAE,CAAC,MAAM,CAAC,IAAI,EAAE,MAAM,EAAE;YAC7C,UAAU,EAAE,KAAK,CAAC,cAAc;YAChC,iBAAiB,EAAE,iBAAiB;YACpC,aAAa;SACd,CAAC,CAAC;QAEH,EAAE;QACF,sBAAsB;QACtB,EAAE;QACF,IAAI,CAAC,WAAW,GAAG,IAAI,EAAE,CAAC,MAAM,CAAC,IAAI,EAAE,MAAM,EAAE;YAC7C,UAAU,EAAE,KAAK,CAAC,cAAc;YAChC,iBAAiB,EAAE,iBAAiB;YACpC,aAAa;SACd,CAAC,CAAC;QACH,IAAI,CAAC,kBAAkB,GAAG,IAAI,EAAE,CAAC,MAAM,CAAC,IAAI,EAAE,SAAS,EAAE;YACvD,UAAU,EAAE,KAAK,CAAC,qBAAqB;YACvC,iBAAiB,EAAE,iBAAiB;YACpC,aAAa;SACd,CAAC,CAAC;QAEH,4BAA4B;QAC5B,IAAI,CAAC,cAAc,GAAG,IAAI,EAAE,CAAC,oBAAoB,CAAC,IAAI,EAAE,KAAK,EAAE;YAC7D,OAAO,EAAE,aAAa,KAAK,SAAS,CAAC,CAAC,CAAC,GAAG,aAAa,GAAG,eAAe,EAAE,CAAC,CAAC,CAAC,SAAS;SACxF,CAAC,CAAC;QACH,IAAI,aAAa,KAAK,SAAS,EAAE;YAC/B,IAAI,CAAC,cAAc,CAAC,kBAAkB,CAAC,aAAa,CAAC,CAAC;SACvD;QAED,4BAA4B;QAC5B,IAAI,CAAC,iBAAiB,GAAG,IAAI,SAAS,CAAC,QAAQ,CAAC,IAAI,CAAC,WAAW,EAAE;YAChE,oBAAoB,EAAE,IAAI,CAAC,aAAa;SACzC,CAAC,CAAC;IACL,CAAC;IAvED,IAAW,UAAU;QACnB,OAAO,IAAI,CAAC,WAAW,CAAC;IAC1B,CAAC;IAGD,IAAW,aAAa;QACtB,OAAO,IAAI,CAAC,cAAc,CAAC;IAC7B,CAAC;IAGD,IAAW,gBAAgB;QACzB,OAAO,IAAI,CAAC,iBAAiB,CAAC;IAChC,CAAC;IAGD,IAAW,iBAAiB;QAC1B,OAAO,IAAI,CAAC,kBAAkB,CAAC;IACjC,CAAC;IAGD,IAAW,UAAU;QACnB,OAAO,IAAI,CAAC,WAAW,CAAC;IAC1B,CAAC;;AAxBH,kCA0EC","sourcesContent":["import { RemovalPolicy } from 'aws-cdk-lib';\nimport * as cf from 'aws-cdk-lib/aws-cloudfront';\nimport * as cforigins from 'aws-cdk-lib/aws-cloudfront-origins';\nimport * as s3 from 'aws-cdk-lib/aws-s3';\nimport { Construct } from 'constructs';\n\n/**\n * Represents a MicroApps S3\n */\nexport interface IMicroAppsS3 {\n  /**\n   * S3 bucket for deployed applications\n   */\n  readonly bucketApps: s3.IBucket;\n\n  /**\n   * CloudFront Origin Access Identity for the deployed applications bucket\n   */\n  readonly bucketAppsOAI: cf.OriginAccessIdentity;\n\n  /**\n   * CloudFront Origin for the deployed applications bucket\n   */\n  readonly bucketAppsOrigin: cforigins.S3Origin;\n\n  /**\n   * S3 bucket for staged applications (prior to deploy)\n   */\n  readonly bucketAppsStaging: s3.IBucket;\n\n  /**\n   * S3 bucket for CloudFront logs\n   */\n  readonly bucketLogs: s3.IBucket;\n}\n\n/**\n * Properties to initialize an instance of `MicroAppsS3`.\n */\nexport interface MicroAppsS3Props {\n  /**\n   * RemovalPolicy override for child resources\n   *\n   * Note: if set to DESTROY the S3 buckets will have `autoDeleteObjects` set to `true`\n   *\n   * @default - per resource default\n   */\n  readonly removalPolicy?: RemovalPolicy;\n\n  /**\n   * S3 deployed apps bucket name\n   *\n   * @default auto-assigned\n   */\n  readonly bucketAppsName?: string;\n\n  /**\n   * S3 staging apps bucket name\n   *\n   * @default auto-assigned\n   */\n  readonly bucketAppsStagingName?: string;\n\n  /**\n   * S3 logs bucket name\n   *\n   * @default auto-assigned\n   */\n  readonly bucketLogsName?: 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/**\n * Create the durable MicroApps S3 Buckets\n *\n * These should be created in a stack that will not be deleted if\n * there are breaking changes to MicroApps in the future.\n */\nexport class MicroAppsS3 extends Construct implements IMicroAppsS3 {\n  private _bucketApps: s3.IBucket;\n  public get bucketApps(): s3.IBucket {\n    return this._bucketApps;\n  }\n\n  private _bucketAppsOAI: cf.OriginAccessIdentity;\n  public get bucketAppsOAI(): cf.OriginAccessIdentity {\n    return this._bucketAppsOAI;\n  }\n\n  private _bucketAppsOrigin: cforigins.S3Origin;\n  public get bucketAppsOrigin(): cforigins.S3Origin {\n    return this._bucketAppsOrigin;\n  }\n\n  private _bucketAppsStaging: s3.IBucket;\n  public get bucketAppsStaging(): s3.IBucket {\n    return this._bucketAppsStaging;\n  }\n\n  private _bucketLogs: s3.IBucket;\n  public get bucketLogs(): s3.IBucket {\n    return this._bucketLogs;\n  }\n\n  constructor(scope: Construct, id: string, props?: MicroAppsS3Props) {\n    super(scope, id);\n\n    if (props === undefined) {\n      throw new Error('props must be set');\n    }\n\n    const { removalPolicy, assetNameRoot, assetNameSuffix } = props;\n\n    // Use Auto-Delete S3Bucket if removal policy is DESTROY\n    const s3AutoDeleteItems = removalPolicy === RemovalPolicy.DESTROY;\n\n    //\n    // S3 Bucket for Logging - Usable by many stacks\n    //\n    this._bucketLogs = new s3.Bucket(this, 'logs', {\n      bucketName: props.bucketLogsName,\n      autoDeleteObjects: s3AutoDeleteItems,\n      removalPolicy,\n    });\n\n    //\n    // S3 Buckets for Apps\n    //\n    this._bucketApps = new s3.Bucket(this, 'apps', {\n      bucketName: props.bucketAppsName,\n      autoDeleteObjects: s3AutoDeleteItems,\n      removalPolicy,\n    });\n    this._bucketAppsStaging = new s3.Bucket(this, 'staging', {\n      bucketName: props.bucketAppsStagingName,\n      autoDeleteObjects: s3AutoDeleteItems,\n      removalPolicy,\n    });\n\n    // Create S3 Origin Identity\n    this._bucketAppsOAI = new cf.OriginAccessIdentity(this, 'oai', {\n      comment: assetNameRoot !== undefined ? `${assetNameRoot}${assetNameSuffix}` : undefined,\n    });\n    if (removalPolicy !== undefined) {\n      this._bucketAppsOAI.applyRemovalPolicy(removalPolicy);\n    }\n\n    // Add Origin for CloudFront\n    this._bucketAppsOrigin = new cforigins.S3Origin(this._bucketApps, {\n      originAccessIdentity: this.bucketAppsOAI,\n    });\n  }\n}\n"]}
@@ -415,5 +415,5 @@ class MicroAppsSvcs extends constructs_1.Construct {
415
415
  }
416
416
  exports.MicroAppsSvcs = MicroAppsSvcs;
417
417
  _a = JSII_RTTI_SYMBOL_1;
418
- MicroAppsSvcs[_a] = { fqn: "@pwrdrvr/microapps-cdk.MicroAppsSvcs", version: "0.2.12" };
418
+ MicroAppsSvcs[_a] = { fqn: "@pwrdrvr/microapps-cdk.MicroAppsSvcs", version: "0.2.13" };
419
419
  //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"MicroAppsSvcs.js","sourceRoot":"","sources":["../src/MicroAppsSvcs.ts"],"names":[],"mappings":"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","sourcesContent":["import { existsSync } from 'fs';\nimport * as path from 'path';\nimport * as apigwy from '@aws-cdk/aws-apigatewayv2-alpha';\nimport * as apigwyAuth from '@aws-cdk/aws-apigatewayv2-authorizers-alpha';\nimport * as apigwyint from '@aws-cdk/aws-apigatewayv2-integrations-alpha';\nimport { Aws, Duration, RemovalPolicy, Stack } from 'aws-cdk-lib';\nimport * as cf from 'aws-cdk-lib/aws-cloudfront';\nimport * as dynamodb from 'aws-cdk-lib/aws-dynamodb';\nimport * as iam from 'aws-cdk-lib/aws-iam';\nimport * as lambda from 'aws-cdk-lib/aws-lambda';\nimport * as lambdaNodejs from 'aws-cdk-lib/aws-lambda-nodejs';\nimport * as logs from 'aws-cdk-lib/aws-logs';\nimport * as s3 from 'aws-cdk-lib/aws-s3';\nimport { Construct } from 'constructs';\n\n/**\n * Properties to initialize an instance of `MicroAppsSvcs`.\n */\nexport interface MicroAppsSvcsProps {\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 for deployed applications\n   */\n  readonly bucketApps: s3.IBucket;\n\n  /**\n   * CloudFront Origin Access Identity for the deployed applications bucket\n   */\n  readonly bucketAppsOAI: cf.OriginAccessIdentity;\n\n  /**\n   * S3 bucket for staged applications (prior to deploy)\n   */\n  readonly bucketAppsStaging: s3.IBucket;\n\n  /**\n   * API Gateway v2 HTTP for Router and app\n   */\n  readonly httpApi: apigwy.HttpApi;\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   * 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   * Path prefix on the root of the deployment\n   *\n   * @example dev/\n   * @default none\n   */\n  readonly rootPathPrefix?: string;\n\n  /**\n   * Require IAM auth on API Gateway\n   *\n   * @default true\n   */\n  readonly requireIAMAuthorization?: boolean;\n\n  /**\n   * Existing table for apps/versions/rules\n   *\n   * @warning - It is *strongly* suggested that production stacks create\n   * their own DynamoDB Table and pass it into this construct, for protection\n   * against data loss due to logical ID changes, the ability to configure\n   * Provisioned capacity with Auto Scaling, the ability to add additional indices, etc.\n   *\n   * Requirements:\n   * - Hash Key: `PK`\n   * - Sort Key: `SK`\n   *\n   * @default created by construct\n   */\n  readonly table?: dynamodb.ITable;\n}\n\n/**\n * Represents a MicroApps Services\n */\nexport interface IMicroAppsSvcs {\n  /**\n   * DynamoDB table used by Router, Deployer, and Release console app\n   */\n  readonly table: dynamodb.ITable;\n\n  /**\n   * Lambda function for the Deployer\n   */\n  readonly deployerFunc: lambda.IFunction;\n\n  /**\n   * Lambda function for the Router\n   */\n  readonly routerFunc: lambda.IFunction;\n}\n\n/**\n * Create a new MicroApps Services construct, including the Deployer\n * and Router Lambda Functions, and the DynamoDB Table used by both.\n */\nexport class MicroAppsSvcs extends Construct implements IMicroAppsSvcs {\n  private _ownedTable?: dynamodb.Table;\n  private _table: dynamodb.ITable;\n  public get table(): dynamodb.ITable {\n    return this._table;\n  }\n\n  private _deployerFunc: lambda.Function;\n  public get deployerFunc(): lambda.IFunction {\n    return this._deployerFunc;\n  }\n\n  private _routerFunc: lambda.Function;\n  public get routerFunc(): lambda.IFunction {\n    return this._routerFunc;\n  }\n\n  constructor(scope: Construct, id: string, props?: MicroAppsSvcsProps) {\n    super(scope, id);\n\n    if (props === undefined) {\n      throw new Error('props cannot be undefined');\n    }\n\n    const {\n      bucketApps,\n      bucketAppsOAI,\n      bucketAppsStaging,\n      s3PolicyBypassAROAs = [],\n      s3PolicyBypassPrincipalARNs = [],\n      s3StrictBucketPolicy = false,\n      appEnv,\n      httpApi,\n      removalPolicy,\n      assetNameRoot,\n      assetNameSuffix,\n      rootPathPrefix = '',\n      requireIAMAuthorization = true,\n    } = props;\n\n    if (s3StrictBucketPolicy === true) {\n      if (s3PolicyBypassAROAs.length === 0 && s3PolicyBypassPrincipalARNs.length === 0) {\n        throw new Error(\n          's3StrictBucketPolicy cannot be true without specifying at least one s3PolicyBypassAROAs or s3PolicyBypassPrincipalARNs',\n        );\n      }\n    }\n\n    //\n    // DynamoDB Table\n    //\n    if (props.table === undefined) {\n      // Create able if none passed\n      this._ownedTable = new dynamodb.Table(this, 'table', {\n        tableName: assetNameRoot ? `${assetNameRoot}${assetNameSuffix}` : undefined,\n        billingMode: dynamodb.BillingMode.PAY_PER_REQUEST,\n        partitionKey: {\n          name: 'PK',\n          type: dynamodb.AttributeType.STRING,\n        },\n        sortKey: {\n          name: 'SK',\n          type: dynamodb.AttributeType.STRING,\n        },\n        removalPolicy,\n      });\n      this._table = this._ownedTable;\n    } else {\n      this._table = props.table;\n    }\n\n    //\n    // Router Lambda Function\n    //\n\n    // Create Router Lambda Function\n    const routerFuncProps: Omit<lambda.FunctionProps, 'handler' | 'code'> = {\n      functionName: assetNameRoot ? `${assetNameRoot}-router${assetNameSuffix}` : undefined,\n      memorySize: 1769,\n      logRetention: logs.RetentionDays.ONE_MONTH,\n      runtime: lambda.Runtime.NODEJS_14_X,\n      timeout: Duration.seconds(15),\n      environment: {\n        NODE_ENV: appEnv,\n        DATABASE_TABLE_NAME: this._table.tableName,\n        AWS_NODEJS_CONNECTION_REUSE_ENABLED: '1',\n        ROOT_PATH_PREFIX: rootPathPrefix,\n      },\n    };\n    if (\n      process.env.NODE_ENV === 'test' &&\n      existsSync(path.join(__dirname, '..', '..', 'microapps-router', 'dist', 'index.js'))\n    ) {\n      // This is for local dev\n      this._routerFunc = new lambda.Function(this, 'router-func', {\n        code: lambda.Code.fromAsset(path.join(__dirname, '..', '..', 'microapps-router', 'dist')),\n        handler: 'index.handler',\n        ...routerFuncProps,\n      });\n    } else if (existsSync(path.join(__dirname, 'microapps-router', 'index.js'))) {\n      // This is for built apps packaged with the CDK construct\n      this._routerFunc = new lambda.Function(this, 'router-func', {\n        code: lambda.Code.fromAsset(path.join(__dirname, 'microapps-router')),\n        handler: 'index.handler',\n        ...routerFuncProps,\n      });\n    } else {\n      // Create Router Lambda Layer\n      const routerDataFiles = new lambda.LayerVersion(this, 'router-templates', {\n        code: lambda.Code.fromAsset(\n          path.join(__dirname, '..', '..', 'microapps-router', 'templates'),\n        ),\n        removalPolicy,\n      });\n\n      this._routerFunc = new lambdaNodejs.NodejsFunction(this, 'router-func', {\n        entry: path.join(__dirname, '..', '..', 'microapps-router', 'src', 'index.ts'),\n        handler: 'handler',\n        bundling: {\n          minify: true,\n          sourceMap: true,\n        },\n        layers: [routerDataFiles],\n        ...routerFuncProps,\n      });\n    }\n    if (removalPolicy !== undefined) {\n      this._routerFunc.applyRemovalPolicy(removalPolicy);\n    }\n    const policyReadTarget = new iam.PolicyStatement({\n      effect: iam.Effect.ALLOW,\n      actions: ['s3:GetObject'],\n      resources: [`${bucketApps.bucketArn}/*`],\n    });\n    for (const router of [this._routerFunc]) {\n      router.addToRolePolicy(policyReadTarget);\n      // Give the Router access to DynamoDB table\n      this._table.grantReadData(router);\n      this._table.grant(router, 'dynamodb:DescribeTable');\n    }\n    // Create alias for Router\n    const routerAlias = this._routerFunc.addAlias('CurrentVersion');\n\n    //\n    // Deployer Lambda Function\n    //\n\n    // Create Deployer Lambda Function\n    const iamRoleUploadName = assetNameRoot\n      ? `${assetNameRoot}-deployer-upload${assetNameSuffix}`\n      : undefined;\n    const deployerFuncName = assetNameRoot\n      ? `${assetNameRoot}-deployer${assetNameSuffix}`\n      : undefined;\n    const deployerFuncProps: Omit<lambda.FunctionProps, 'handler' | 'code'> = {\n      functionName: deployerFuncName,\n      memorySize: 1769,\n      logRetention: logs.RetentionDays.ONE_MONTH,\n      runtime: lambda.Runtime.NODEJS_14_X,\n      timeout: Duration.seconds(15),\n      environment: {\n        NODE_ENV: appEnv,\n        APIGWY_ID: httpApi.httpApiId,\n        DATABASE_TABLE_NAME: this._table.tableName,\n        FILESTORE_STAGING_BUCKET: bucketAppsStaging.bucketName,\n        FILESTORE_DEST_BUCKET: bucketApps.bucketName,\n        AWS_NODEJS_CONNECTION_REUSE_ENABLED: '1',\n        ROOT_PATH_PREFIX: rootPathPrefix,\n        REQUIRE_IAM_AUTHORIZATION: requireIAMAuthorization ? 'true' : 'false',\n      },\n    };\n    if (\n      process.env.NODE_ENV === 'test' &&\n      existsSync(path.join(__dirname, '..', '..', 'microapps-deployer', 'dist', 'index.js'))\n    ) {\n      // This is for local dev\n      this._deployerFunc = new lambda.Function(this, 'deployer-func', {\n        code: lambda.Code.fromAsset(path.join(__dirname, '..', '..', 'microapps-deployer', 'dist')),\n        handler: 'index.handler',\n        ...deployerFuncProps,\n      });\n    } else if (existsSync(path.join(__dirname, 'microapps-deployer', 'index.js'))) {\n      // This is for built apps packaged with the CDK construct\n      this._deployerFunc = new lambda.Function(this, 'deployer-func', {\n        code: lambda.Code.fromAsset(path.join(__dirname, 'microapps-deployer')),\n        handler: 'index.handler',\n        ...deployerFuncProps,\n      });\n    } else {\n      this._deployerFunc = new lambdaNodejs.NodejsFunction(this, 'deployer-func', {\n        entry: path.join(__dirname, '..', '..', 'microapps-deployer', 'src', 'index.ts'),\n        handler: 'handler',\n        bundling: {\n          minify: true,\n          sourceMap: true,\n        },\n        ...deployerFuncProps,\n      });\n    }\n    if (removalPolicy !== undefined) {\n      this._deployerFunc.applyRemovalPolicy(removalPolicy);\n    }\n    // Give the Deployer access to DynamoDB table\n    this._table.grantReadWriteData(this._deployerFunc);\n    this._table.grant(this._deployerFunc, 'dynamodb:DescribeTable');\n\n    //\n    // Deloyer upload temp role\n    // Deployer assumes this role with a limited policy to generate\n    // an STS temp token to return to microapps-publish for the upload.\n    //\n    const iamRoleUpload = new iam.Role(this, 'deployer-upload-role', {\n      roleName: iamRoleUploadName,\n      inlinePolicies: {\n        uploadPolicy: new iam.PolicyDocument({\n          statements: [\n            new iam.PolicyStatement({\n              actions: ['s3:ListBucket'],\n              resources: [bucketAppsStaging.bucketArn],\n            }),\n            new iam.PolicyStatement({\n              actions: ['s3:PutObject', 's3:GetObject', 's3:AbortMultipartUpload'],\n              resources: [`${bucketAppsStaging.bucketArn}/*`],\n            }),\n          ],\n        }),\n      },\n      assumedBy: this._deployerFunc.grantPrincipal,\n    });\n    this._deployerFunc.addEnvironment('UPLOAD_ROLE_NAME', iamRoleUpload.roleName);\n\n    //\n    // Update S3 permissions\n    //\n    // Create PrincipalARN List\n    const s3PolicyBypassArnPrincipals: iam.ArnPrincipal[] = [];\n    for (const arnPrincipal of s3PolicyBypassPrincipalARNs) {\n      s3PolicyBypassArnPrincipals.push(new iam.ArnPrincipal(arnPrincipal));\n    }\n    // Create AROA List that matches assumed sessions\n    const s3PolicyBypassAROAMatches: string[] = [];\n    for (const aroa of s3PolicyBypassAROAs) {\n      s3PolicyBypassAROAMatches.push(`${aroa}:*`);\n    }\n    // Deny apps from reading:\n    // - If they are missing the microapp-name tag\n    // - Anything outside of the folder that matches their microapp-name tag\n    const policyDenyPrefixOutsideTag = new iam.PolicyStatement({\n      sid: 'deny-prefix-outside-microapp-name-tag',\n      effect: iam.Effect.DENY,\n      actions: ['s3:*'],\n      notPrincipals: [\n        new iam.CanonicalUserPrincipal(\n          bucketAppsOAI.cloudFrontOriginAccessIdentityS3CanonicalUserId,\n        ),\n        new iam.AccountRootPrincipal(),\n        ...s3PolicyBypassArnPrincipals,\n        this._deployerFunc.grantPrincipal,\n      ],\n      notResources: [\n        `${bucketApps.bucketArn}/\\${aws:PrincipalTag/microapp-name}/*`,\n        bucketApps.bucketArn,\n      ],\n      conditions: {\n        Null: { 'aws:PrincipalTag/microapp-name': 'false' },\n        // StringNotLike: {'aws:'}\n      },\n    });\n    if (removalPolicy !== undefined) {\n      policyDenyPrefixOutsideTag.addCondition(\n        // Allows the DeletableBucket Lambda to delete items in the buckets\n        'StringNotLike',\n        { 'aws:PrincipalTag/application': `${Stack.of(this).stackName}-core*` },\n      );\n    }\n    const policyDenyMissingTag = new iam.PolicyStatement({\n      sid: 'deny-missing-microapp-name-tag',\n      effect: iam.Effect.DENY,\n      actions: ['s3:*'],\n      notPrincipals: [\n        new iam.CanonicalUserPrincipal(\n          bucketAppsOAI.cloudFrontOriginAccessIdentityS3CanonicalUserId,\n        ),\n        new iam.AccountRootPrincipal(),\n        // Exclude the Deployer Function directly\n        this._deployerFunc.grantPrincipal,\n        // 2021-12-04 - Not 100% sure that this is actually needed...\n        // Let's test this and remove if actually not necessary\n        new iam.ArnPrincipal(\n          `arn:aws:sts::${Aws.ACCOUNT_ID}:assumed-role/${this._deployerFunc.role?.roleName}/${this._deployerFunc.functionName}`,\n        ),\n        ...s3PolicyBypassArnPrincipals,\n      ],\n      resources: [`${bucketApps.bucketArn}/*`, bucketApps.bucketArn],\n      conditions: {\n        Null: { 'aws:PrincipalTag/microapp-name': 'true' },\n        // Note: This AROA must be specified to prevent this policy from locking\n        // out non-root sessions that have assumed the admin role.\n        // The notPrincipals will only match the role name exactly and will not match\n        // any session that has assumed the role since notPrincipals does not allow\n        // wildcard matches and does not do them implicitly either.\n        // The AROA must be used because there are only 3 Principal variables:\n        //  https://docs.aws.amazon.com/IAM/latest/UserGuide/reference_policies_variables.html#principaltable\n        //  aws:username, aws:userid, aws:PrincipalTag\n        // For an assumed role, aws:username is blank, aws:userid is:\n        //  [unique id AKA AROA for Role]:[session name]\n        // Table of unique ID prefixes such as AROA:\n        //  https://docs.aws.amazon.com/IAM/latest/UserGuide/reference_identifiers.html#identifiers-prefixes\n        // The name of the role is simply not available and if it was\n        // we'd need to write a complicated comparison to make sure\n        // that we didn't exclude the Deny tag from roles in other accounts.\n        //\n        // To get the AROA with the AWS CLI:\n        //   aws iam get-role --role-name ROLE-NAME\n        //   aws iam get-user -–user-name USER-NAME\n        StringNotLike: { 'aws:userid': [Aws.ACCOUNT_ID, ...s3PolicyBypassAROAMatches] },\n      },\n    });\n    if (removalPolicy !== undefined) {\n      policyDenyMissingTag.addCondition(\n        // Allows the DeletableBucket Lambda to delete items in the buckets\n        'StringNotLike',\n        { 'aws:PrincipalTag/application': `${Stack.of(this).stackName}-core*` },\n      );\n    }\n    const policyCloudFrontAccess = new iam.PolicyStatement({\n      sid: 'cloudfront-oai-access',\n      effect: iam.Effect.ALLOW,\n      actions: ['s3:GetObject', 's3:ListBucket'],\n      principals: [\n        new iam.CanonicalUserPrincipal(\n          bucketAppsOAI.cloudFrontOriginAccessIdentityS3CanonicalUserId,\n        ),\n      ],\n      resources: [`${bucketApps.bucketArn}/*`, bucketApps.bucketArn],\n    });\n\n    if (bucketApps.policy === undefined) {\n      const document = new s3.BucketPolicy(this, 's3-policy', {\n        bucket: bucketApps,\n      }).document;\n      document.addStatements(policyCloudFrontAccess);\n\n      if (s3StrictBucketPolicy) {\n        document.addStatements(policyDenyPrefixOutsideTag);\n        document.addStatements(policyDenyMissingTag);\n      }\n    } else {\n      bucketApps.policy.document.addStatements(policyCloudFrontAccess);\n\n      if (s3StrictBucketPolicy) {\n        bucketApps.policy.document.addStatements(policyDenyPrefixOutsideTag);\n        bucketApps.policy.document.addStatements(policyDenyMissingTag);\n      }\n    }\n\n    // Allow the Lambda to read from the staging bucket\n    const policyReadListStaging = new iam.PolicyStatement({\n      effect: iam.Effect.ALLOW,\n      // FIXME: Allow Deployer to delete from Staging bucket\n      actions: ['s3:DeleteObject', 's3:GetObject', 's3:ListBucket'],\n      resources: [`${bucketAppsStaging.bucketArn}/*`, bucketAppsStaging.bucketArn],\n    });\n    this._deployerFunc.addToRolePolicy(policyReadListStaging);\n\n    // Allow the Lambda to write to the target bucket and delete\n    const policyReadWriteListTarget = new iam.PolicyStatement({\n      effect: iam.Effect.ALLOW,\n      actions: ['s3:DeleteObject', 's3:GetObject', 's3:PutObject', 's3:ListBucket'],\n      resources: [`${bucketApps.bucketArn}/*`, bucketApps.bucketArn],\n    });\n    this._deployerFunc.addToRolePolicy(policyReadWriteListTarget);\n\n    // Allow the deployer to get a temporary STS token\n    const policyGetSTSToken = new iam.PolicyStatement({\n      effect: iam.Effect.ALLOW,\n      actions: ['sts:GetFederationToken'],\n      resources: ['*'],\n    });\n    this._deployerFunc.addToRolePolicy(policyGetSTSToken);\n\n    // Allow the deployer to assume the upload role\n    const policyAssumeUpload = new iam.PolicyStatement({\n      effect: iam.Effect.ALLOW,\n      actions: ['sts:AssumeRole'],\n      resources: [iamRoleUpload.roleArn],\n    });\n    this._deployerFunc.addToRolePolicy(policyAssumeUpload);\n\n    //\n    // Give Deployer permissions to create routes and integrations\n    // on the API Gateway API.\n    //\n\n    // Grant the ability to List all APIs (we have to find it)\n    const policyAPIList = new iam.PolicyStatement({\n      effect: iam.Effect.ALLOW,\n      actions: ['apigateway:GET'],\n      resources: [`arn:aws:apigateway:${Aws.REGION}::/apis`],\n    });\n    this._deployerFunc.addToRolePolicy(policyAPIList);\n    // Grant full control over the API we created\n    const policyAPIManage = new iam.PolicyStatement({\n      effect: iam.Effect.ALLOW,\n      actions: ['apigateway:*'],\n      resources: [\n        `arn:aws:apigateway:${Aws.REGION}:${Aws.ACCOUNT_ID}:${httpApi.httpApiId}/*`,\n        `arn:aws:apigateway:${Aws.REGION}::/apis/${httpApi.httpApiId}/integrations/*`,\n        `arn:aws:apigateway:${Aws.REGION}::/apis/${httpApi.httpApiId}/integrations`,\n        `arn:aws:apigateway:${Aws.REGION}::/apis/${httpApi.httpApiId}/routes`,\n        `arn:aws:apigateway:${Aws.REGION}::/apis/${httpApi.httpApiId}/routes/*`,\n      ],\n    });\n    this._deployerFunc.addToRolePolicy(policyAPIManage);\n    // Grant full control over lambdas that indicate they are microapps\n    const policyAPIManageLambdas = new iam.PolicyStatement({\n      effect: iam.Effect.ALLOW,\n      actions: ['lambda:*'],\n      resources: [\n        `arn:aws:lambda:${Aws.REGION}:${Aws.ACCOUNT_ID}:function:*`,\n        `arn:aws:lambda:${Aws.REGION}:${Aws.ACCOUNT_ID}:function:*:*`,\n      ],\n      conditions: {\n        StringEqualsIfExists: { 'aws:ResourceTag/microapp-managed': 'true' },\n      },\n    });\n    this._deployerFunc.addToRolePolicy(policyAPIManageLambdas);\n\n    // This creates an integration and a router\n    const route = new apigwy.HttpRoute(this, 'route-default', {\n      httpApi,\n      routeKey: apigwy.HttpRouteKey.DEFAULT,\n      integration: new apigwyint.HttpLambdaIntegration('router-integration', routerAlias),\n      authorizer: requireIAMAuthorization ? new apigwyAuth.HttpIamAuthorizer() : undefined,\n    });\n\n    let routeArn = route.routeArn;\n    // Remove the trailing `/` on the ARN, which is not correct\n    if (routeArn.endsWith('/')) {\n      routeArn = routeArn.slice(0, routeArn.length - 1);\n    }\n\n    // Grant API Gateway permission to invoke the Lambda\n    new lambda.CfnPermission(this, 'router-invoke', {\n      action: 'lambda:InvokeFunction',\n      functionName: this._routerFunc.functionName,\n      principal: 'apigateway.amazonaws.com',\n      sourceArn: routeArn,\n    });\n  }\n}\n"]}
@@ -38,5 +38,5 @@ class MicroAppsTable extends constructs_1.Construct {
38
38
  }
39
39
  exports.MicroAppsTable = MicroAppsTable;
40
40
  _a = JSII_RTTI_SYMBOL_1;
41
- MicroAppsTable[_a] = { fqn: "@pwrdrvr/microapps-cdk.MicroAppsTable", version: "0.2.12" };
41
+ MicroAppsTable[_a] = { fqn: "@pwrdrvr/microapps-cdk.MicroAppsTable", version: "0.2.13" };
42
42
  //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiTWljcm9BcHBzVGFibGUuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvTWljcm9BcHBzVGFibGUudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7QUFDQSxxREFBcUQ7QUFDckQsMkNBQXVDO0FBd0R2Qzs7R0FFRztBQUNILE1BQWEsY0FBZSxTQUFRLHNCQUFTO0lBTTNDLFlBQVksS0FBZ0IsRUFBRSxFQUFVLEVBQUUsS0FBMkI7UUFDbkUsS0FBSyxDQUFDLEtBQUssRUFBRSxFQUFFLENBQUMsQ0FBQztRQUVqQixJQUFJLEtBQUssS0FBSyxTQUFTLEVBQUU7WUFDdkIsTUFBTSxJQUFJLEtBQUssQ0FBQywyQkFBMkIsQ0FBQyxDQUFDO1NBQzlDO1FBRUQsTUFBTSxFQUFFLGFBQWEsRUFBRSxhQUFhLEVBQUUsZUFBZSxFQUFFLEdBQUcsS0FBSyxDQUFDO1FBRWhFLEVBQUU7UUFDRixpQkFBaUI7UUFDakIsRUFBRTtRQUNGLElBQUksQ0FBQyxNQUFNLEdBQUcsSUFBSSxRQUFRLENBQUMsS0FBSyxDQUFDLElBQUksRUFBRSxPQUFPLEVBQUU7WUFDOUMsU0FBUyxFQUFFLGFBQWEsQ0FBQyxDQUFDLENBQUMsR0FBRyxhQUFhLEdBQUcsZUFBZSxFQUFFLENBQUMsQ0FBQyxDQUFDLFNBQVM7WUFDM0UsV0FBVyxFQUFFLFFBQVEsQ0FBQyxXQUFXLENBQUMsZUFBZTtZQUNqRCxZQUFZLEVBQUU7Z0JBQ1osSUFBSSxFQUFFLElBQUk7Z0JBQ1YsSUFBSSxFQUFFLFFBQVEsQ0FBQyxhQUFhLENBQUMsTUFBTTthQUNwQztZQUNELE9BQU8sRUFBRTtnQkFDUCxJQUFJLEVBQUUsSUFBSTtnQkFDVixJQUFJLEVBQUUsUUFBUSxDQUFDLGFBQWEsQ0FBQyxNQUFNO2FBQ3BDO1lBQ0QsYUFBYTtTQUNkLENBQUMsQ0FBQztJQUNMLENBQUM7SUE3QkQsSUFBVyxLQUFLO1FBQ2QsT0FBTyxJQUFJLENBQUMsTUFBTSxDQUFDO0lBQ3JCLENBQUM7O0FBSkgsd0NBZ0NDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgUmVtb3ZhbFBvbGljeSB9IGZyb20gJ2F3cy1jZGstbGliJztcbmltcG9ydCAqIGFzIGR5bmFtb2RiIGZyb20gJ2F3cy1jZGstbGliL2F3cy1keW5hbW9kYic7XG5pbXBvcnQgeyBDb25zdHJ1Y3QgfSBmcm9tICdjb25zdHJ1Y3RzJztcblxuLyoqXG4gKiBQcm9wZXJ0aWVzIHRvIGluaXRpYWxpemUgYW4gaW5zdGFuY2Ugb2YgYE1pY3JvQXBwc1RhYmxlYC5cbiAqL1xuZXhwb3J0IGludGVyZmFjZSBNaWNyb0FwcHNUYWJsZVByb3BzIHtcbiAgLyoqXG4gICAqIFJlbW92YWxQb2xpY3kgb3ZlcnJpZGUgZm9yIGNoaWxkIHJlc291cmNlc1xuICAgKlxuICAgKiBOb3RlOiBpZiBzZXQgdG8gREVTVFJPWSB0aGUgUzMgYnVja2VzIHdpbGwgaGF2ZSBgYXV0b0RlbGV0ZU9iamVjdHNgIHNldCB0byBgdHJ1ZWBcbiAgICpcbiAgICogQGRlZmF1bHQgLSBwZXIgcmVzb3VyY2UgZGVmYXVsdFxuICAgKi9cbiAgcmVhZG9ubHkgcmVtb3ZhbFBvbGljeT86IFJlbW92YWxQb2xpY3k7XG5cbiAgLyoqXG4gICAqIEFwcGxpY2F0aW9uIGVudmlyb25tZW50LCBwYXNzZWQgYXMgYE5PREVfRU5WYFxuICAgKiB0byB0aGUgUm91dGVyIGFuZCBEZXBsb3llciBMYW1iZGEgZnVuY3Rpb25zXG4gICAqL1xuICByZWFkb25seSBhcHBFbnY6IHN0cmluZztcblxuICAvKipcbiAgICogT3B0aW9uYWwgYXNzZXQgbmFtZSByb290XG4gICAqXG4gICAqIEBleGFtcGxlIG1pY3JvYXBwc1xuICAgKiBAZGVmYXVsdCAtIHJlc291cmNlIG5hbWVzIGF1dG8gYXNzaWduZWRcbiAgICovXG4gIHJlYWRvbmx5IGFzc2V0TmFtZVJvb3Q/OiBzdHJpbmc7XG5cbiAgLyoqXG4gICAqIE9wdGlvbmFsIGFzc2V0IG5hbWUgc3VmZml4XG4gICAqXG4gICAqIEBleGFtcGxlIC1kZXYtcHItMTJcbiAgICogQGRlZmF1bHQgbm9uZVxuICAgKi9cbiAgcmVhZG9ubHkgYXNzZXROYW1lU3VmZml4Pzogc3RyaW5nO1xuXG4gIC8qKlxuICAgKiBQYXRoIHByZWZpeCBvbiB0aGUgcm9vdCBvZiB0aGUgZGVwbG95bWVudFxuICAgKlxuICAgKiBAZXhhbXBsZSBkZXYvXG4gICAqIEBkZWZhdWx0IG5vbmVcbiAgICovXG4gIHJlYWRvbmx5IHJvb3RQYXRoUHJlZml4Pzogc3RyaW5nO1xufVxuXG4vKipcbiAqIFJlcHJlc2VudHMgYSBNaWNyb0FwcHNUYWJsZVxuICovXG5leHBvcnQgaW50ZXJmYWNlIElNaWNyb0FwcHNUYWJsZSB7XG4gIC8qKlxuICAgKiBEeW5hbW9EQiB0YWJsZSB1c2VkIGJ5IFJvdXRlciwgRGVwbG95ZXIsIGFuZCBSZWxlYXNlIGNvbnNvbGUgYXBwXG4gICAqL1xuICByZWFkb25seSB0YWJsZTogZHluYW1vZGIuVGFibGU7XG59XG5cbi8qKlxuICogQ3JlYXRlIGEgbmV3IE1pY3JvQXBwcyBUYWJsZSBmb3IgYXBwcyAvIHZlcnNpb25zIC8gcnVsZXNcbiAqL1xuZXhwb3J0IGNsYXNzIE1pY3JvQXBwc1RhYmxlIGV4dGVuZHMgQ29uc3RydWN0IGltcGxlbWVudHMgSU1pY3JvQXBwc1RhYmxlIHtcbiAgcHJpdmF0ZSBfdGFibGU6IGR5bmFtb2RiLlRhYmxlO1xuICBwdWJsaWMgZ2V0IHRhYmxlKCk6IGR5bmFtb2RiLlRhYmxlIHtcbiAgICByZXR1cm4gdGhpcy5fdGFibGU7XG4gIH1cblxuICBjb25zdHJ1Y3RvcihzY29wZTogQ29uc3RydWN0LCBpZDogc3RyaW5nLCBwcm9wcz86IE1pY3JvQXBwc1RhYmxlUHJvcHMpIHtcbiAgICBzdXBlcihzY29wZSwgaWQpO1xuXG4gICAgaWYgKHByb3BzID09PSB1bmRlZmluZWQpIHtcbiAgICAgIHRocm93IG5ldyBFcnJvcigncHJvcHMgY2Fubm90IGJlIHVuZGVmaW5lZCcpO1xuICAgIH1cblxuICAgIGNvbnN0IHsgcmVtb3ZhbFBvbGljeSwgYXNzZXROYW1lUm9vdCwgYXNzZXROYW1lU3VmZml4IH0gPSBwcm9wcztcblxuICAgIC8vXG4gICAgLy8gRHluYW1vREIgVGFibGVcbiAgICAvL1xuICAgIHRoaXMuX3RhYmxlID0gbmV3IGR5bmFtb2RiLlRhYmxlKHRoaXMsICd0YWJsZScsIHtcbiAgICAgIHRhYmxlTmFtZTogYXNzZXROYW1lUm9vdCA/IGAke2Fzc2V0TmFtZVJvb3R9JHthc3NldE5hbWVTdWZmaXh9YCA6IHVuZGVmaW5lZCxcbiAgICAgIGJpbGxpbmdNb2RlOiBkeW5hbW9kYi5CaWxsaW5nTW9kZS5QQVlfUEVSX1JFUVVFU1QsXG4gICAgICBwYXJ0aXRpb25LZXk6IHtcbiAgICAgICAgbmFtZTogJ1BLJyxcbiAgICAgICAgdHlwZTogZHluYW1vZGIuQXR0cmlidXRlVHlwZS5TVFJJTkcsXG4gICAgICB9LFxuICAgICAgc29ydEtleToge1xuICAgICAgICBuYW1lOiAnU0snLFxuICAgICAgICB0eXBlOiBkeW5hbW9kYi5BdHRyaWJ1dGVUeXBlLlNUUklORyxcbiAgICAgIH0sXG4gICAgICByZW1vdmFsUG9saWN5LFxuICAgIH0pO1xuICB9XG59XG4iXX0=
@@ -5,6 +5,6 @@
5
5
  <title></title>
6
6
  </head>
7
7
  <body>
8
- <iframe src="{{iframeSrc}}" seamless height="100%" width="100%" frameBorder="0"></iframe>
8
+ <iframe src="{{iframeSrc}}" seamless frameBorder="0" style="top: 0; left: 0; width: 100%; height: 100%; position: fixed"></iframe>
9
9
  </body>
10
10
  </html>
package/package.json CHANGED
@@ -83,7 +83,7 @@
83
83
  "publishConfig": {
84
84
  "access": "public"
85
85
  },
86
- "version": "0.2.12",
86
+ "version": "0.2.13",
87
87
  "types": "lib/index.d.ts",
88
88
  "stability": "experimental",
89
89
  "jsii": {
package/releasetag.txt CHANGED
@@ -1 +1 @@
1
- v0.2.12
1
+ v0.2.13
package/version.txt CHANGED
@@ -1 +1 @@
1
- 0.2.12
1
+ 0.2.13