gh-stack


I use this tool to help managed stacked pull requests on Github, which are notoriously difficult to manage manually. Here are a few examples of the kind of things I'm talking about:

This tool assumes that all PRs in a single "stack" all have a unique identifier in their title (I typically use a Jira ticket number for this). It then looks for all PRs containing this containing this identifier and builds a dependency graph in memory. This can technically support a "branched stack" instead of a single chain, but I haven't really tried the latter style. Note that the gh-stack rebase command will definitely not work with the branched style.

With this graph built up, the tool can:

Usage

```bash $ export GHSTACKOAUTHTOKEN=''

Idempotently add a markdown table summarizing the stack

to the description of each PR in the stack.

$ gh-stack github 'stack-identifier'

Same as above, but precede the markdown table with the

contents of filename.txt.

$ gh-stack github 'stack-identifier' filename.txt

Print a description of the stack to stdout.

$ gh-stack log 'stack-identifier'

Emit a bash script that can update a stack in the case of conflicts.

WARNING: This script could potentially cause destructive behavior.

$ gh-stack rebase 'stack-identifier' ```

Strategy

Here's a quick summary of the strategy that the bash script described above uses to keep the stack up-to-date.

Let's use this stack as an example:

Feature Changes

In the first case, let's assume that "feature part 1" had some changes added to it in the form of a commit; this leaves parts 2 & 3 in a conflicted state:

The script requires that you pass in a PREBASE ref (which is essentially the boundary for the feature part you're operating on - in this case the parent of the last/oldest commit in feature-part-2). The script starts cherry-picking commits at this ref for the first iteration. An initial TO ref is also required, which is the point upon which you want to rebase the rest of the stack. In this case, that ref is remote/feature-part-1).

The script executes a single step, we now have this intermediate state:

The script completes execution, and we now have this final state with the entire stack updated/recreated:

Feature Complete & Merged

In the second case, let's assume that "feature part 2" is done and has been merged to develop:

This immediately leaves feature parts 2 & 3 in a conflicted state. The script can fix this situation as well. As before, pass a PREBASE (in this case the parent of the oldest commit in feature part 2) and an initial TO ref to rebase on (in this case remote/develop).

Once the script executes a single step, we're left with:

And once the script is done:

Disclaimer

Use at your own risk (and make sure your git repository is backed up), especially because: