name: stack-merge
description: Bulk-merge a Graphite stack into main via admin-bypass fast-forward push. Replaces the Graphite merge queue for rivet-dev/rivet. Invoke when the user says "merge the stack at ", "ship everything through ", "bulk-merge up to ", "fast-forward main to ", or any request to land multiple stacked PRs at once in one shot. Requires repo admin permissions (bypasses branch protection), Graphite (gt) CLI, and GitHub (gh) CLI. After the FF push, closes all in-scope PRs explicitly with gh pr close in parallel — GitHub does not auto-mark them as MERGED because their bases point at sibling stack branches, not main.
gh pr close in parallel — GitHub does not auto-mark them as MERGED because their bases point at sibling stack branches, not main.Stack Merge
Land an entire Graphite downstack into main in one admin-bypass fast-forward push, then close every in-scope PR explicitly. No per-PR CI gates, no merge commits on main, no gh pr merge per PR. PRs end up in Closed state (not MERGED) unless their base was already main, but their commits are in main's history.
This skill replaces Graphite's merge queue — use when you want the outcome Graphite's queue produces (linear history, all PRs merged) but without waiting for the queue. Requires admin bypass on main branch protection.
When NOT to use
- User wants normal single-PR merge → use
gh pr merge. - User isn't a repo admin → push will reject (tell them to use Graphite's queue).
- Stack tip has failing CI and user cares → queue would refuse; pause here too.
Preconditions (verify at start; bail early)
- Admin/bypass on main:
gh api repos/rivet-dev/rivet/branches/main/protection— check user is inbypass_pull_request_allowances.users[]or has repo admin. If not, stop. - Graphite initialized:
gt lssucceeds. - Clean working tree:
git status --shortempty. In-flight rebase/merge state must be resolved first. - Worktree sanity:
git worktree list— prune anyprunableentries withgit worktree prune. Stale worktrees break rebases with "already checked out at" errors. See references/gotchas.md.
The flow (8 phases)
Each phase has a validation gate or confirmation gate. Writes happen only after the user explicitly approves the final commands.
Phase 1 — Scope
Input: target branch name from user.
- Enumerate the merge path: walk
gh pr view <branch> --json baseRefNamefrom target down tomain. Produces the ordered list of all PRs that will be closed in Phase 8. - Do not infer parents from
gt lsvisual order — use each PR'sbaseRefName. See gotchas.md. - Script:
scripts/list_merge_path.sh <target-branch>.
Phase 2 — Sync
gt sync --no-interactive. Pulls latest main, cleans up merged upstream branches.- Read output for any
(needs restack)markers in the merge path.
Phase 3 — Unfreeze
- List frozen branches in the merge path (
gt ls | grep frozen). gt syncsilently skips frozen branches. If any frozen branch sits between target and main, the chain stays anchored to its old main SHA forever. See gotchas.md.- Unfreeze each one:
gt unfreeze <branch>. Operation is metadata-only, no rewrites yet. Present full list to user and confirm before running.
Phase 4 — Restack downstack
gt restack --downstackfrom the target branch. Cascades rebase from main up through the (now-unfrozen) chain to target.- Conflicts: hand off to user (
gt continueafter resolving each). Do not drive conflict resolution. - When restack completes, verify:
git merge-base --is-ancestor origin/main <target>must return YES. If NO, main moved during the restack —gt syncagain and loop.
Phase 5 — Validation gate (read-only)
Script: scripts/validate.sh <target-branch>. Checks:
- FF safety:
git merge-base --is-ancestor origin/main origin/<target>→ must be YES. - Commits on main not in target:
git rev-list --count origin/<target>..origin/main→ must be 0 (otherwise FF push deletes commits on main). - Local vs remote divergence for every branch in the merge path (restack rewrote SHAs; remote needs updating).
- Commit hygiene (informational, not blocking):
- Ralph-style commits (pattern:
^[a-z]+(\(.+?\))?!?:\s*\[?US-\d+\]?\s*-\s). See scripts/detect_ralph.sh. - Unsquashed branches (>1 commit against baseRefName).
- Ralph-style commits (pattern:
- Conflict preview via scratch worktree +
git merge origin/main --no-commit --no-ff. Shouldn't conflict after a clean restack, but if it does, loop back to Phase 4.
Gate: if any check fails, stop and present findings.
Phase 6 — Confirmation gate
Present to user before any writes:
- main SHA before/after
- number of commits landing
- list of PRs that will be closed in Phase 8 (the merge path)
- list of branches that will be force-pushed
Require explicit "yes". No proceed-on-silence.
Phase 7 — Push
Two writes, in order:
- Batch force-push all merge-path branches. Each branch's remote head must match its restacked local SHA so the PR's head-SHA-on-GitHub reflects the post-restack commits. This matters only if restack actually rewrote SHAs; with a clean sync and no-op restack it's a harmless no-op.
Single-transaction multi-refspec push is faster than N sequential pushes. Usegit push origin --force-with-lease \ <branch1> <branch2> ... <target>--force-with-lease, never--force. - FF push main to target tip.
Admin bypass will be noted in push output ("Bypassed rule violations for refs/heads/main").git push origin origin/<target>:main
Phase 8 — Close PRs + cleanup
- Close every open PR whose head is now reachable from main, in parallel. This catches both the in-scope PRs and anything below the merge path that also landed.
PRs whose base wasmain_sha=$(git rev-parse origin/main) open_prs=$(gh pr list --state open --limit 200 --json number,headRefOid \ --jq '.[] | "\(.number)\t\(.headRefOid)"') close=() while IFS=$'\t' read -r n sha; do git merge-base --is-ancestor "$sha" "$main_sha" 2>/dev/null && close+=("$n") done <<<"$open_prs" for n in "${close[@]}"; do (gh pr close "$n" --comment "Landed in main via stack-merge fast-forward push. Commits are in main; closing to match." &) done waitmainmay have auto-merged by now —gh pr closeprintsalready closedfor those, which is fine. gt syncto clean up locally-merged branches.- Report final state: main SHA, count of PRs closed, any PRs still OPEN (shouldn't be any).
Why explicit close: GitHub only auto-closes a PR as MERGED when head ⊆ the PR's base branch. Most stacked PRs have base = sibling branch, not main, so the cheap auto-merge check doesn't fire. GitHub's slower background sweep eventually closes them as CLOSED (not MERGED), but it's unreliable and can take 5+ minutes. Close them ourselves.
References
- references/gotchas.md — every trap from the flow's design session. Read this before writing any push commands.
- references/phases.md — per-phase detail, example outputs, decision trees.
Scripts
scripts/list_merge_path.sh <target>— enumerate in-scope PRs via baseRefName traversal.scripts/validate.sh <target>— run all Phase 5 read-only checks.scripts/detect_ralph.sh <target>— flag Ralph-style commits in the merge path.scripts/exec_merge.sh <target> --confirm— batch force-push + FF push (Phase 7). Requires explicit--confirmflag to execute.
Core rules
- Never skip the FF safety check.
git merge-base --is-ancestor origin/main origin/<target>must be YES before the main push. If commits exist on main that aren't in target, the push deletes them. gt lsvisual ordering is not PR parent ordering. Always usegh pr view <N> --json baseRefName.baseRefNamecan be either a real branch name orgraphite-base/<pr#>— both are valid.gt syncsilently skips frozen branches. Freeze is opt-out-of-restack. Any frozen branch between target and main breaks the merge path.- Use
--force-with-leasefor all force-pushes. Bails safely if remote moved;--forceblindly overwrites. - Hand off conflicts during
gt restack. Do not pick sides. The user resolves, then runsgt continue. - Close PRs explicitly in Phase 8, don't wait for auto-close. GitHub only auto-closes as MERGED when head ⊆ base branch, which almost never holds for stacked PRs. See gotchas.md.
- Phase 8 closes everything reachable from main, not just
list_merge_path.shoutput. That script stops atgraphite-base/<N>boundaries and can miss PRs below.