A serverless form submission application, built and deployed to AWS with the Serverless Application Framework. This QuickStart app provides a template for deploying your own QuickStart codebase. The Architecture Diagram shows the resources built by the QuickStart template in this repo.
A service-specific README is located in each service folder (services/).
Note: serverless-stack.com is different from serverless.com. This app does not use serverless-stack.com.
You'll need an AWS account with appropriate IAM permissions (Admin is recommended) to deploy this app to AWS.
This is useful for getting insight into your resources. Instructions for installing and upgrading the AWS CLI are here.
Use Node Version Manager (NVM) to install and manage Node. Instructions to install NVM are here. Inspect the install script before invoking it with bash.
The file .nvmrc
contains the Node version that is expected to be used with this app. This version matches the Lambda runtime. Use NVM to install this version before running the app.
# from the root project directory
$ nvm -v # verify nvm is installed
$ nvm install # install Node.js version specificed in .nvmrc
$ nvm use
$ nvm list # verify the expected Node.js version is active
This app uses Yarn as the package manager for Node. Use the Node package manager included with Node (NPM) to install Yarn. Instructions for installing Yarn are here. This link includes instructions for installing on Windows.
# installation on MacOS and Linux
$ npm install -g yarn # use to install or upgrade Yarn
The app is built on a framework called Serverless (for AWS). Instructions for installing the Serverless CLI, using AWS as the cloud provider, are here: Serverless Getting Started page
# installation on MacOS and Linux
$ npm install -g serverless # use to install or upgrade serverless
$ serverless -v # displays version
$ serverless -h # displays help menu
In addition to the pre-requisite tools above, Java must be installed locally to support the local deployment of DynamoDB. (The downloadable version of DynamoDB is an executable .jar file. To run DynamoDB locally, you must have the Java Runtime Environment (JRE) version 8.x or newer.
Local dev is configured in typescript project in ./src
. The entry point is ./src/dev.ts
, it manages running the moving pieces locally: the API, the database, the filestore, and the frontend.
Local dev is built around the Serverless plugin serverless-offline
. serverless-offline
runs an API gateway locally, configured by ./services/app-api/serverless.yml
and hot-reloads your lambdas on every save. The plugins serverless-dynamodb-local
and serverless-s3-local
stand up the local DynamoDB and local S3 in a similar fashion.
When run locally, auth bypasses Cognito. The frontend mimics login using local storage with a mock user, and sends an id in the cognito-identity-id
header on every request. serverless-offline
expects that and sets it as the cognitoId in the requestContext for your lambdas, just like Cognito would in AWS.
- valid short-term access keys for your AWS account pasted in your terminal window
Run all the services locally with the command ./scripts/dev.sh local
. For example,
# from root directory of project
$ ./scripts/dev.sh local
Note: This repo does not support selecting a subset of services to run locally.
You can view the app on localhost:3000.
- valid short-term access keys for your AWS account pasted in your terminal window
- a QuickStart stage, , deployed and running in AWS
The QuickStart contains a script services/ui-src/configure_local.sh
that configures the frontend to run locally and connect with the backend services running in AWS.
You can view the app on localhost:3000.
# ensure that <stage> has already been deployed # This is necessary since only the frontend is deployed locally
$ aws cloudformation list-stacks | grep <stage> # confirm the stage is deployed
# change to services/ui-src directory
$ cd services/ui-src
$ ./configure_local.sh <stage> # sets environment variables
$ yarn run start # start React frontend
# login using valid credentials
# to stop frontend, hit Control-C
./scripts/dev.sh test
runs tests against locally deployed services. This is just a placeholder implementation and should be replaced with an actual implementation in src/dev.ts
$ ./scripts/dev.sh test
"Testing 1. 2. 3."
- valid short-term access keys for your AWS account pasted in your terminal window
- stage names must be less than 27 characters
# from root directory of project
$ ./scripts/deploy.sh <stage>
After the deployment completes, verify all resources have been deployed. From the AWS console, navigate to CloudFormation. Filter on the stage name. From the command line:
$ aws cloudformation list-stacks | grep <stage>
To destroy AWS resources for a stage:
# from root directory of project
$ ./scripts/destroy.sh <stage>
$ aws cloudformation list-stacks | grep <stage> # This should return no matches.
Note: Parameters created in AWS Systems Manager (SSM) parameter store are not destroyed.
The Quickstart contains the following types of tests:
These are component-based tests defined in the service folders in /services
. They use
- Jest as a test runner
- the React Testing Library to facilitate querying the DOM in the same way the user would
- API mocks (see Amendments.test.js for an example of how to mock an API endpoint for a component-based test)
Unit tests can be run manually with the scripts/unit_test.sh
script, and they are run automatically as part of the deploy
GitHub Action
These tests are defined in the tests/cypress
folder. They use
For more info, including how to run these tests locally, see the testing README.
Integration and accessibility tests are run automatically as part of the deploy
GitHub Action.
Our product is promoted through branches. A developer branch is merged to the master branch. The master branch is merged to val to affect a val release, and the val branch is merged to production to affect a production release. Please use the buttons below to promote/release code to higher environments.
branch | status | release |
---|---|---|
master | ||
val | ||
production |
Deployments via GitHub Actions authenticate using Open ID Connect (OIDC). This prevents needing to maintain a separate static user within AWS for the deployments.
-
For each AWS environment, create a parameters file with the settings needed for your deployments. Example files are provided in .github/oidc/.
- The permissions boundary must match the one needed for your environment.
- The list of IAM policies (custom and AWS-managed) must provide your role with sufficient permissions (Note: multiple policies can be provided within the comma-separated list).
- SubjectClaimFilters depend on your environment setup. The provided examples assume that you wish to allow any branch to be able to deploy to the dev AWS environment, but only want the
val
andproduction
branches to be able to deploy to their respective AWS environments. For more options on setting the subject claim filters, see About security hardening with OpenID Connect: Example subject claims.
-
Setup the OIDC role in each AWS environment using the provided CloudFormation template and the AWS CLI.
# verify the AWS account you are authenticated to $ aws sts get-caller-identity $ aws cloudformation help # for help with command parameters # view the changes to AWS resources using the flag `--no-execute-changeset` $ aws cloudformation deploy --template-file .github/oidc/github-actions-oidc-template.yml --stack-name github-actions-oidc-role --parameter-overrides file://PATH_TO_ENVIRONMENT.json --capabilities CAPABILITY_IAM --no-execute-changeset # verify changes; if OK, deploy changes $ aws cloudformation deploy --template-file .github/oidc/github-actions-oidc-template.yml --stack-name github-actions-oidc-role --parameter-overrides file://PATH_TO_ENVIRONMENT.json --capabilities CAPABILITY_IAM
-
In GitHub Secrets => Actions, create a repository secret for each role that a job must assume. The secret allows a job to get credentials for a given AWS environment. A job accesses a secret using dot notation:
secrets.SECRET_NAME
. The workflows in .github/workflows reference the DEV secret name (secrets.AWS_OIDC_ROLE_TO_ASSUME).
AWS Environment | Secret Name | Secret Value (Service Role ARN) |
---|---|---|
DEV | AWS_OIDC_ROLE_TO_ASSUME | |
VAL | VAL_AWS_OIDC_ROLE_TO_ASSUME | |
PROD | PRODUCTION_AWS_OIDC_ROLE_TO_ASSUME |
See current open issues or check out the project board.
Please feel free to open new issues for defects or enhancements.
To contribute:
- Fork this repository
- Make changes in your fork
- Open a pull request targeting this repository
Pull requests are being accepted.
See LICENSE for full details.
As a work of the United States Government, this project is
in the public domain within the United States.
Additionally, we waive copyright and related rights in the
work worldwide through the CC0 1.0 Universal public domain dedication.
To enable slack integration, set a value for SLACK_WEBHOOK_URL in GitHub actions secret.
To set the SLACK_WEBHOOK_URL:
- Go to https://api.slack.com/apps
- Create a new app
- Fill in the information
- Select
Add features and functionality
- Select
Incoming Webhooks
- Enable
Activate Incoming Webooks
- Select
Add New Webhook to Workspace
- Create a secret in GitHub Actions Secrets called SLACK_WEBHOOK_URL and set its value to the new webhook URL
SLACK_WEBHOOK CREATION (NEW Process):
- Put in a Jira ticket
- Go to https://jira.cms.gov/servicedesk/customer/portal/1
- Select I need help with Jira, Confluence, Slack, or GitHub. You need the following EUA Job Code to access OC Jira: JIRA_Users
- If you are able to login to https://jira.cms.gov/servicedesk/customer/portal/1/create/11 you are good to go.
- One of the tools team member will create the Slack App and provide the webhook information back to ADO's.
Please join the macpro-quickstart-serverless slack channel to get all build status and also contribute to any ongoing discussions. Join here: https://join.slack.com/t/macproquickst-ugp3045/shared_invite/zt-mdxpbtkk-SrLRi_yzJrXX3uYgvrbjlg
Mike Dial |
Seth Sacher |
---|