Thanks for your interest in contributing to the RFDK! ❤️
This document describes how to set up a development environment and submit your contributions. Please read it carefully and let us know if it's not up-to-date (even better, submit a PR with your corrections ;-)).
The RFDK is written in Typescript and converted, using jsii, into Python. Thus, the minimal development environment must include:
- Node.js >= 18.0.0
- docker >= 18
We also recommend developing on a Linux system.
To set up a local Node.js environment, we suggest using the Node Version Manager. Follow the instructions to install it into your system. Once nvm is installed then you can install a version of Node.js and set your shell to make it available when you login:
# For example, the latest version of Node.js 18.x
# Find out the version number for latest
LATEST_VERSION=$(nvm ls-remote | grep v18 | grep 'Latest' | awk '{print $1}')
# Install it
nvm install ${LATEST_VERSION}
# Set your shell to use that version when starting (for bash)
echo "nvm use --delete-prefix ${LATEST_VERSION}" >> ${HOME}/.bashrc
# or for zsh
echo "nvm use --delete-prefix ${LATEST_VERSION}" >> ${HOME}/.zshrc
# Alternatively, just enter the Node.js environment when you want.
nvm use --delete-prefix ${LATEST_VERSION}
Once you have set up Node.js, then you will want to set up the Yarn package manager, and Typescript:
npm install -g yarn typescript
The official installation instructions are great. We suggest following the post-installation steps as well for convenience.
It is recommended you perform the packaging step of the build process within a docker container on a Linux-compatible system. The developers actively use Linux for development, but macOS and the Windows Subsystem for Linux #2 may also work.
-
Build the RFDK and ensure that all unit tests pass:
# From the root directory of this repository ./build.sh # Once you've run ./build.sh from the root directory at least once, then # you can do subsequent build & test iterations from the RFDK package directory to save time. cd packages/aws-rfdk yarn build+test
-
When you're ready to package the build artifacts into installable packages (e.g. For running integration tests, using with an example app, or for using with your own app) then first enter the docker container:
cd <directory containing this file> # This will also do a docker pull of the required container image # the first time that you run it. ./scripts/rfdk_build_environment.sh >>> bash-4.2#
-
Now, to create the npm and python packages:
# From within the docker container at the root directory of this repository >>> bash-4.2# ./pack.sh # The resulting artifacts will be located in the dist/ directory.
The script ./link-all.sh
can be used to generate symlinks to all modules in this repository under some node_module
directory. This can be used to develop against this repo as a local dependency.
One can use the postinstall
script to symlink this repo. For example:
{
"scripts": {
"postinstall": "../AWS-RFDK/link-all.sh"
}
}
This assumes this repo is a sibling of the target repo and will install the CDK as a linked dependency during yarn install.
The packages that are created by the ./pack.sh
script can be installed directly into an npm or Python environment.
# Installing into a local npm environment
npm install <RFDK directory>/dist/js/[email protected]
# Installing into a Python environment
pip install --force <RFDK directory>/dist/python/aws-rfdk-0.15.0.tar.gz
If you are switching git branches and start to encounter unexpected build errors, then the cause may be stale build artifacts from a previous build from another branch.
We have scripts that attempt to identify stale build artifacts and clean them up. You might try those.
# Ensure that git is tracking all of the active changes that you care about.
git add <files>
# Run the cleaning scripts
cd <this directory>
yarn clean
./clean.sh
# Rebuild
./build.sh
If that does not work, then you might try the following nuclear option. WARNING -- this will delete all files in the workspace that are not actively tracked by git:
# Ensure that git is tracking all of the active changes that you care about.
git add <files>
# Use git clean to delete all files that are not tracked in git
git clean -fdx
- Testing
- Unit test added (prefer not to modify an existing test, otherwise, it's probably a breaking change)
- Docs
-
jsdocs: All public APIs documented
-
README: README and/or documentation topic updated
-
Architecture Diagrams: Any new constructs with non-trivial complexity should have accompanying architectural diagrams. Similarly, any changes to a construct's architecture should also be reflected in the architecture diagrams.
Please read the architecture diagrams README for more details.
-
Breaking?: add upgrade documentation to
packages/aws-rfdk/docs/upgrade
which clearly guides users through the process of assessing whether they are impacted by the breaking change and to determine what action they should take to upgrade
-
- Title and Description
- Change type: title prefixed with fix, feat and module name in parens, which will appear in changelog
- Title: use lower-case and doesn't end with a period
- Breaking?: last paragraph: "BREAKING CHANGE: <describe what changed + link for details>"
- Issues: Indicate issues fixed via: "Fixes #xxx" or "Closes #xxx"
If there isn't one already, open an issue describing what you intend to contribute. It's useful to communicate in advance, because sometimes, someone is already working in this space, so maybe it's worth collaborating with them instead of duplicating the efforts.
In some cases, it is useful to seek for feedback by iterating on a design document. This is useful when you plan a big change or feature, or you want advice on what would be the best path forward.
Sometimes, the GitHub issue is sufficient for such discussions, and can be sufficient to get clarity on what you plan to do. Sometimes, a design document would work better, so people can provide iterative feedback.
In such cases, use the GitHub issue description to collect requirements and use cases for your feature.
Work your magic. Here are some guidelines:
- Coding style (abbreviated):
- In general, follow the style of the code around you
- 2 space indentation
- 120 characters wide
- ATX style headings in markdown (e.g.
## H2 heading
)
- Every change requires a unit test
- If you change APIs, make sure to update the README file
- Try to maintain a single feature/bugfix per pull request. It's okay to introduce a little bit of housekeeping changes along the way, but try to avoid conflating multiple features. Eventually all these are going to go into a single commit, so you can use that to frame your scope.
- If your change introduces a new construct, then take a look at our existing constructs to get a feel for the common patterns that we use.
We have two styles of test in the RFDK.
- Unit tests.
- These are found in the
test/
directories withinpackages/aws-rfdk
. They essentially act as regression detectors. - In the tests for constructs, we verify that they are generating the expected CloudFormation resources, with expected attributes. This allows us to detect a change in behavior when we make a change, and to identify changes in our upstream dependencies that materially alter our generated resources.
- The tests for our AWS Lambda functions similarly act as proof that the function acts as expected, and are used to detect unintended side-effects of changes made to the code.
- Generally, we aim for our tests to cover as close to 100% of code paths as possible, and to verify that all construct properties correctly materialize in the resulting CloudFormation template.
- These are found in the
- Functional/integration tests.
- These are found in the
integ/
directory of the root of the repository. - These tests help us detect when changes to the RFDK, or to the software that we deploy, causes regressions in the functionality of the farm.
- These tests are best-effort coverage. Minimally, we aim for them to cover the core use-cases and any regressions that have been found and fixed in the past -- to reduce the change of a re-regression.
- These are found in the
Create a commit with the proposed changes:
-
Commit title and message (and PR title and description) must adhere to conventionalcommits.
- The title must begin with
feat(module): title
,fix(module): title
,refactor(module): title
orchore(module): title
. Our module titles are:core
-- for code related to constructs underpackages/aws-rfdk/lib/core
.deadline
-- for code related to constructs underpackages/aws-rfdk/lib/deadline
.aws-rfdk
-- for code underpackages/aws-rfdk
that does not fall into one of the above categories.integ
-- for changes to the integration tests underinteg/
.examples
-- for changes to our example applications underexamples/
.tools
-- for changes to our tooling undertools/
.lambda-layers
-- for changes to the lambda-layer code underlambda-layers/
repo
-- for all other changes.
- Title should be lowercase.
- No period at the end of the title.
- The title must begin with
-
Commit message should describe motivation. Think about your code reviewers and what information they need in order to understand what you did. If it's a big commit (hopefully not), try to provide some good entry points so it will be easier to follow.
-
Commit message should indicate which issues are fixed:
fixes #<issue>
orcloses #<issue>
. -
Shout out to collaborators.
-
If not obvious (i.e. from unit tests), describe how you verified that your change works.
-
If this commit includes breaking changes, they must be listed at the end in the following format (notice how multiple breaking changes should be formatted):
BREAKING CHANGE: Description of what broke and how to achieve this behavior now
- **module-name:** Another breaking change
- **module-name:** Yet another breaking change
- Push to a personal GitHub fork.
- Submit a Pull Request on GitHub. A reviewer will later be assigned by the maintainers.
- Please follow the PR checklist written above. We trust our contributors to self-check, and this helps that process!
- Discuss review comments and iterate until you get at least one “Approve”. When iterating, push new commits to the same branch. Usually all these are going to be squashed when you merge to master. The commit messages should be hints for you when you finalize your merge commit message.
- Make sure to update the PR title/description if things change. The PR title/description are going to be used as the commit title/message and will appear in the CHANGELOG, so maintain them all the way throughout the process.
- Make sure your PR builds successfully (we have a github action set up to automatically build all PRs)
- Once approved and tested, a maintainer will squash-merge to master and will use your PR title/description as the commit message.
Current best practices is to always used a fixed version for dependencies — "aws-rfdk": "1.0.0"
.
Unfortunately allowing any kind of nonfixed type) dependencies causes build errors. CDK is using fixed dependencies for all their packages so we have no reasonable way to allow nonfixed dependencies as well.
If you want to learn more about dependencies you can read the yarn docs.
You will need to put the package's dependencies in the dependencies
and
peerDependencies
. For example, if you wanted to use aws-sqs,
you would need to have this in the package.json
# packages/aws-rfdk/package.json
"dependencies": {
"@aws-cdk/aws-sqs": "1.18.0"
},
"peerDependencies": {
"@aws-cdk/aws-sqs": "1.18.0"
},
For each dependency you specified, you will need to include its dependencies in
the root package.json
. For example, if you wanted to use aws-sqs,
you would need to have this in your root package.json
"devDependencies": {
"@aws-cdk/aws-cloudwatch": "1.18.0",
"@aws-cdk/aws-iam": "1.18.0",
"@aws-cdk/aws-kms": "1.18.0",
"@aws-cdk/core": "1.18.0"
}