Stack Attack
Stack Attack is a CLI tool that helps you work with stacked pull requests.
Stacking PRs is now a piece many small pieces of cake.
Demo video:
data:image/s3,"s3://crabby-images/c545a/c545aab7d98bd64a345ca9f8749966bd73bb1235" alt="Hackathon demo video"
Problem
Although big PRs are common in the GitHub ecosystem, there are many problems
with them:
- Big PRs are hard to review as it is hard to follow the many logic changes in a
large diff. As the code review process is not going to be as effective, there
is a higher chance that bugs get merged and deployed into production.
- Big PRs take some time to write, potentially causing integration issues both
on the base branch and in the PR as the project evolves around it.
Are small PRs the solution? YES.
However, small PRs are difficult to deal with on GitHub. Some issues include:
- Small PRs result in a faster PR cadence. Without an easy and systematic way to
build PRs on each other (while the PR author waits for a PR review) and manage
their dependencies, the PRs quickly become unmanageable.
- Updates to earlier PRs or the base branch will require many individual rebases
of every PR in the dependency stack.
Stacked PRs?
Instead of implementing a feature in a single large PR, Stack Attack lets you
build a feature incrementally in a stack of PRs. Specifically, Stack Attack
helps you to create, manipulate, and land these PR stacks.
data:image/s3,"s3://crabby-images/484f5/484f5103344ae3d149c523719f5014222f662ec9" alt="Many stacked PRs"
Stack Attack is inspired by Facebook's internal source management and IDE GUI
tools, and implements their workflow for everyone else.
Principles
-
A stacked PR has exactly one commit. A commit can only get to a certain
size before it gets confusing and unmanageable to even you. This is a good
thing – if a diff is too big for you, splitting it into multiple commits
solves the problem for you but not for the reviewer.
-
A stacked PR implements one small thing completely. When a PR has a
single, well-defined goal, code reviewers can more easily review your code,
allowing you to land your code faster.
-
A stacked PR is fully functional and only depends on its base branch and
other PRs stacked below it. In other words, a partial PR stack should be
able to be landed without breaking the base branch. This allows early PRs in
the PR stack to be landed while work on their dependent PRs continue, allowing
all developers to detect integration issues early.
data:image/s3,"s3://crabby-images/702a4/702a4eb945d29e830f27e334ec34bd300d4651c9" alt="Weaving PR stacks onto master"
Limitations
- As a feature is now implemented across multiple PRs, it can be hard to get an
overview of what the PR stack achieves. PRs are also no longer a suitable
place for higher level discussions. Discussions that occur in a PR early in a
stack will also likely not be seen or remembered by those reviewing later PRs.
These can be mitigated by moving high level discussions to other tools such as
GitHub Issues.
- Reviewers can no longer consider a PR individually, but they have to also
consider the changes in later PRs stacked on top of it. If done inefficiently,
this can lead to unnecessary comments on earlier PRs that already have
solutions in later PRs, or long review processes that review the stack as a
whole (and defeating the point of using stacked PRs). This can be mitigated by
the following:
- Adding TODO comments in earlier PRs where later code is expected to go.
- Clearly defining the scope of the PR in its description.
- Highlighting known issues and plans to fix them.
- Rebase conflicts are still a thing. This can be mitigated by reviewing PRs
promptly and landing PRs frequently. Project maintainers may also consider
allowing contributors to address review comments in a new PR on top of the
stack.
- Stack Attack only works with Git and GitHub. (Sorry!)
Non-goals
- Stacked PRs do not replace single PRs. There are many good reasons to use
single PRs, e.g. for small or automated tasks. What Stack Attack gives you is
the additional ability to use stacked PRs when the situation calls for it.
Usage
From NPM
-
Add a <repo_path>/sttack.config.json
file, where <repo_path>
is the local
path to the repository you want to run Stack Attack on. The config file
should contain the following:
{
"personalAccessToken": "<your GitHub personal access token>",
// Provide an SSH key so that we can push your code to GitHub.
"userPublicKeyPath": "<absolute path to your SSH public key>", // e.g.: /Users/phuonganh/.ssh/id_rsa.pub
"userPrivateKeyPath": "<absolute path to your SSH private key>", // e.g.: /Users/phuonganh/.ssh/id_rsa"
"userPassphrase": "<SSH key passphrase>" // Leave empty if no passphrase
}
Note: You can generate a GitHub personal access token
here.
-
Run npx sttack <repo_path>
to start Stack Attack.
From Source
- Clone this repository.
- Run
yarn
to install our dependencies. - Create a
<repo_path>/sttack.config.json
file as above. - Run
yarn cli <repo_path>
to start Stack Attack.
General Tips
- Roughly plan your implementation steps before starting to code. It is much
harder to split a completed feature into a PR stack than to incrementally
build up a PR stack.
- Create and land PRs frequently to get early reviewer feedback. Although you
can update a PR early in the stack and rebase later PRs on top of it, you'll
still want to minimize the chance of facing conflicts when rebasing.
- Review PRs starting from the base of the stack. When working with a large PR
stack, it can be frustrating to have all PRs approved except for the very
first one, blocking the entire stack from being landed.
- In many situations, it may help to gate the new code behind a feature flag
while it is being worked on. This lets you land incomplete features (e.g. a
modal with just placeholder text) safely without breaking the app in
production.
Workflows
Below are some common workflows that Stack Attack supports.
As Stack Attack is in an early phase of development, many essential features are
not yet implemented. Nevertheless, to communicate our vision for Stack Attack,
we've listed these unimplemented workflows together with those that are
implemented. Please reach out if you'd like to help make them a reality!
Creating a PR stack
- Navigate to the base of the commit stack.
- Press
s
to create the PR stack.
data:image/s3,"s3://crabby-images/499cd/499cda90c966dd3b88519450ee705827adc6748d" alt="Create stack UI"
A PR will be created for each commit in the stack.
Navigating a PR stack on GitHub
Every PR references other PRs in the the same stack.
data:image/s3,"s3://crabby-images/417dd/417dde9dc440a381cf536b53ec106e7f95f9a099" alt="Stacked PRs"
Protip: Use commit naming conventions to help you identify the bigger
feature the stack is building towards. At Facebook, Diffs belonging to stacks
are often numbered. Here are some common patterns:
[<large feature name>][<PR number in the stack>/<total number of PRs in the stack, or just "n">] <commit title>
<large feature name> (<PR number in the stack>/<total number of PRs in the stack, or just "n">): <commit title>
<large feature name> (Part <PR number in the stack>)
Some examples of such numbered commits:
- [Input form redesign][1/n] Create new fork of existing code
- V2 data migration (3/n): Remove V1 types
- Add type annotations (Part 1)
Numbered PRs in the wild:
data:image/s3,"s3://crabby-images/eff45/eff450d00834869c2cd9940555401f3a33c42579" alt="A stack of PRs"
Rebasing commits belonging to a PR stack
In a manually-managed PR stack, rebasing is tedious as every PR's branch needs
to be rebased individually.
Stack Attack treats rebasing as a transplanting of a commit tree. As such, with
Stack Attack, you can rebase a stack of commits in just a few keyboard commands.
data:image/s3,"s3://crabby-images/ec019/ec01905964a1c9993328c3b246dd00703888bc0d" alt="Rebasing a stack of PRs"
- Navigate to the base of the commit tree you want to rebase/transplant.
- Press
r
to begin rebasing. - Navigate to the target commit you want to rebase the commit stack on.
- Press
c
to confirm the rebase. The commits in the stack will be rebased,
and all the local branches pointing to them will be moved along. - (Envisioned) If PRs have already been created for the previous commits, press
s
to update the PRs in the PR stack.
Amending PRs in a PR stack (envisioned)
- Checkout the commit to be amended.
- Make your changes.
- In Stack Attack, press
a
to amend the commit. All the commits stacked on
top of it will be rebased onto the new commit. - If PRs have already been created for the previous commits, press
s
to make
a new PR for the PR stack.
Adding PRs on top of an existing PR stack (envisioned)
- Create the new commit at the top of the commit stack using your regular Git
tools.
- In Stack Attack, navigate to the new commit.
- Press
s
to make a new PR for the commit.
Inserting a new PR into an existing PR stack (envisioned)
- Checkout a commit you want to build on.
- Make your changes, then create a new commit using your regular Git tools.
- In Stack Attack, rebase the later commits in the stack onto the new commit.
- Navigate to the new commit.
- Press
s
to make a new PR for the commit.
Landing a complete PR stack (envisioned)
TODO
Landing part of a PR stack (envisioned)
TODO
Inspiration
Built With
- Node.js :computer: - A JavaScript runtime built on
Chrome's V8 JavaScript engine.
- Git :smile: - Version Control Management
- TypeScript :heart: - Semantics
Contributors
Thanks goes to these wonderful people: