Voicebox triage-prs
Use this skill to triage the open PR queue before a release. Classifies every open PR into must-merge, candidate, superseded, or deferred; writes a working triage doc; and runs the merge loop end-to-end. Designed for the pre-release "PR speedrun" pass where a solo maintainer wants to clear the inbound backlog in a single session.
git clone https://github.com/jamiepine/voicebox
T=$(mktemp -d) && git clone --depth=1 https://github.com/jamiepine/voicebox "$T" && mkdir -p ~/.claude/skills && cp -r "$T/.agents/skills/triage-prs" ~/.claude/skills/jamiepine-voicebox-triage-prs && rm -rf "$T"
.agents/skills/triage-prs/SKILL.mdTriage PRs
Goal
Turn a backlog of open PRs into a shipped set of merges in a single focused session. Produce a tracked, resumable plan (
<VERSION>_PR_TRIAGE.md), then work it — rebasing where needed, merging in isolation-safe batches, applying post-merge follow-ups, and closing superseded or partially-applicable PRs with credit to their authors.
This skill pairs with
draft-release-notes and release-bump: triage first, then draft notes against the new main, then cut the release.
When to use
- Before a minor or major release when 10+ open PRs have accumulated
- When you want to unblock merging without losing the narrative of what's landing
- When you know you can't personally review every PR deeply, but need to land the critical subset fast
Prerequisites
CLI authenticated against the repogh- A dedicated worktree for PR review (avoid contaminating
with checkouts of contributor branches)main - Clarity on the target version — the triage doc is named after it (e.g.
)0.4.0_PR_TRIAGE.md
Workflow
1. Set up an isolated PR-review worktree
git worktree list # check for stale ones first git worktree prune git worktree add ../voicebox-pr-review -b pr-review-<VERSION> main
Keep the main worktree for release-prep work (changelog drafts, direct-to-main follow-ups). Keep the review worktree for
gh pr checkout — each checkout moves HEAD to a contributor branch, which you don't want to do in the main worktree.
2. Gather metadata for every open PR
gh pr list --state open --limit 50 --json \ number,title,author,isDraft,mergeable,mergeStateStatus,files,additions,deletions,reviewDecision,statusCheckRollup,maintainerCanModify \ --jq '.[] | {num: .number, title, author: .author.login, mergeable, state: .mergeStateStatus, canModify: .maintainerCanModify, changes: "+\(.additions)/-\(.deletions)", files: [.files[].path]}'
You want, for each PR:
- Size (
)+additions/-deletions - Mergeable state (
,CLEAN
,UNSTABLE
= conflicts,DIRTY
= GitHub still computing)UNKNOWN - Whether maintainer edits are allowed on the branch (needed later if you rebase for the author)
- File paths touched (helps spot overlaps between PRs)
UNKNOWN is common right after a push to main — just try the merge and see.
3. Classify into tiers
Sort each PR into exactly one bucket:
Tier 1 — Merge: small, mergeable, fixes a real bug, clean CI, low review cost. One-liners, dependency relaxations, targeted safety hardening. These are the easy wins.
Tier 2 — Candidate, review: medium size (50-200 lines), touches more surface area, looks sound but needs a closer read. New user-facing features that fit the product direction.
Supersede: the fix or feature is already covered by something merged. Close with a comment pointing to the superseding PR. Check carefully — "similar title" isn't proof; compare the actual diffs.
Defer to next release: big features, dirty conflicts, draft PRs, anything touching the release pipeline in ways that would introduce risk. Don't merge these in a speedrun — they need dedicated focus.
4. Write the triage doc
Create
<VERSION>_PR_TRIAGE.md in the PR-review worktree root. Structure:
# <Repo> <VERSION> — PR Triage Working doc for tracking which open PRs land in <VERSION>. Delete after release cut. Last updated: <DATE> ## Progress **Tier 1: 0 / N merged** **Tier 2: 0 / M handled** **Supersede triage: pending** --- ## Merge for <VERSION> — critical bug fixes | PR | Status | Size | What it fixes | Why must-have | |---|---|---|---|---| | [#123](url) | [ ] | +5/-0 | ... | ... | ## Strong candidate — needs a quick review | PR | Status | Size | Summary | |---|---|---|---| ## Close as superseded | PR | Status | Reason | |---|---|---| ## Defer to <NEXT_VERSION> - [#xxx](url) ... — reason --- ## Order of attack 1. Close superseded PRs (one-liner comments) 2. Merge tier-1 in dependency-free batches — check file paths don't overlap 3. Review tier-2 individually 4. Rerun `draft-release-notes` to pick up everything 5. Run `release-bump`
The Progress header is the most important part — it's your scoreboard and lets you resume cleanly if the session gets interrupted.
5. Work the loop — per PR
For each PR in the tier-1 / tier-2 list:
a. Checkout in the review worktree:
cd ../voicebox-pr-review git checkout pr-review-<VERSION> # reset to neutral base gh pr checkout <N>
b. Read the actual commit, not
:main..HEAD
git show HEAD # the PR's actual changes git show --stat HEAD # files touched + line counts
Do NOT review via
if the PR branch is older than main. That diff includes every commit that landed on main after the PR was forked as git diff main..HEAD
- (deletion) lines. A 3-line PR can look like a 700-line revert. This is the single easiest way to misjudge a PR.
c. Evaluate concerns: correctness, scope, interaction with already-merged work, version compatibility (e.g. can't use an API that requires a dependency version we don't yet pin).
d. Rebase if the branch is behind main:
git fetch origin main git rebase origin/main
This is essential before squash-merging. GitHub's squash computes
diff(PR-head, merge-base) — on a stale branch, that diff includes reverting every in-between commit. Rebasing moves the merge-base forward so the squash is clean.
e. If maintainer edits are allowed, push the rebase back to the contributor's fork:
git remote add <author> https://github.com/<author>/<repo>.git git fetch <author> <branch> # get their ref first git push <author> HEAD:<branch> --force-with-lease
This keeps GitHub's PR UI in sync with the rebased state and makes the merge clean from the GitHub side.
f. Merge:
gh pr merge <N> --squash
g. Update the triage doc — flip the checkbox to
✅ merged <sha> (use the short SHA from gh pr view <N> --json mergeCommit --jq '.mergeCommit.oid[0:7]'). Update the Progress header.
6. Batch tiny fixes
PRs with ≤5 line changes, clean CI, non-overlapping file paths, and obviously-correct intent (e.g. one-line dependency relax, env var add, import path fix) can be merged in a single loop without the review-per-PR ceremony:
for pr in 425 384 416 429; do echo "=== Merging PR $pr ===" gh pr merge $pr --squash done
Verify afterward that each landed cleanly:
for pr in 425 384 416 429; do gh pr view $pr --json state,mergeCommit --jq "{pr: $pr, state, sha: .mergeCommit.oid[0:7]}" done
7. Post-merge follow-ups
Sometimes a PR is worth merging despite a known minor issue (e.g. incomplete dtype map, stale sentinel cleanup). Don't block the merge; apply the follow-up as a normal branch + PR right after:
cd <main-worktree> git pull --ff-only origin main git checkout -b fix/<short-name> # edit... git commit -m "fix(<area>): <one-liner>" git push -u origin fix/<short-name> gh pr create --title "..." --body "Follow-up to #<N>. ..."
Record both SHAs in the triage doc (
✅ merged <pr-sha> + follow-up <pr>).
Direct-to-main exception: only under an explicit, scoped policy (e.g. "release speedrun"). Don't default to it.
8. Supersede: close with a credit-pointing comment
gh pr close <N> --comment "Closing — superseded by merged #<M> which landed <brief description>. Thanks!"
Check the diffs first — "similar title" is not enough. If the PR is partially superseded (the diagnosis is right but only half the changes are still needed), do a partial-apply instead.
9. Partial-apply pattern
When a PR has both valuable and questionable changes bundled:
cd <main-worktree> git pull --ff-only origin main # Cherry-pick specific files from the PR branch git checkout <pr-commit-sha> -- <file1> <file2> # Review the staged changes, adjust as needed git diff --cached # Apply any surgical edits to files you don't want to bulk-replace # (e.g. the PR's file predates a recent main commit you need to preserve) # Commit with a trailer crediting the original author git commit -m "$(cat <<'EOF' <subject> <body explaining what was kept vs dropped> Co-Authored-By: <author> <noreply@github.com> EOF )" git push ... # branch + PR, unless under the direct-to-main exception
Then close the PR with a comment explaining what was applied and what was dropped, referencing the commit SHA.
10. Keep the doc current
Every merge, every close, every follow-up → update
<VERSION>_PR_TRIAGE.md. The doc is your session log. If you're interrupted and resume tomorrow, the doc is the only source of truth for "where am I."
11. When triage is done
- Every PR in the doc has a terminal status (✅ merged / ✅ closed / deferred)
- Progress header shows N/N for each tier
- Next skill to run is
(to regeneratedraft-release-notes
against the new main), then[Unreleased]release-bump
You can delete the triage doc after the release ships, or keep it in version history as a record.
Gotchas
on a stale branch lies. It shows everything main gained since the branch split as deletions. Always review viamain..HEAD
for the PR's actual commit.git show HEAD- Squash-merging an unrebased branch reverts in-between work. The squash computes
. Rebase moves the merge-base forward.diff(PR-head, merge-base)
is transient — GitHub is recomputing after a push. Just try the merge.mergeable=UNKNOWN- Route ordering matters (FastAPI and similar):
must be registered beforeDELETE /history/failed
, or the parameterized path will consumeDELETE /history/{id}
as an ID."failed" - Apple's
overrides-weak_framework
for the same framework, regardless of order — use it via-framework
when a dependency hard-links something optional.cargo:rustc-link-arg=-Wl,-weak_framework,Name - Dependency version floors constrain what you can apply. Before accepting a kwarg rename like
→torch_dtype=
, check the min-version pin supports it. Sometimes the right move is to cherry-pick half the PR.dtype=
and similarcpal::Stream
audio types can't cross!Send
points orawait
. Sometimes a "not-ideal but correct" sync wait is the best available fix; flag but don't block.spawn_blocking- PyTorch nightly builds are not shippable for releases — non-deterministic, can regress between runs. If a PR suggests switching to nightly to fix a GPU issue, prefer
or wait for stable support instead.TORCH_CUDA_ARCH_LIST=...+PTX
Canonical commands reference
# Bulk PR metadata gh pr list --state open --limit 50 --json number,title,author,mergeable,mergeStateStatus,additions,deletions,maintainerCanModify,files # Detailed single-PR view gh pr view <N> --json body,author,headRefName,baseRefName,mergeable,maintainerCanModify,files,statusCheckRollup # The actual commit, not the branch-vs-main diff git show HEAD git show --stat HEAD gh pr diff <N> # Rebase contributor branch onto current main git fetch origin main && git rebase origin/main # Push rebase back to contributor fork (maintainerCanModify=true required) git remote add <author> https://github.com/<author>/<repo>.git git fetch <author> <branch> git push <author> HEAD:<branch> --force-with-lease # Merge gh pr merge <N> --squash # Confirm merge SHA for triage doc gh pr view <N> --json state,mergeCommit --jq '{state, sha: .mergeCommit.oid[0:7]}' # Close superseded gh pr close <N> --comment "Closing — superseded by merged #<M>. Thanks!"
Notes
- Never review a stale branch via
. This is the single most important line in this skill.main..HEAD - The triage doc is the session state. Lose the doc, lose the session. Update it after every action.
- Credit contributors even on partial-applies. Use
trailers and close comments that link to the applied commit.Co-Authored-By: - Don't let perfect be the enemy of shipped. A fix that goes from "broken" to "works with a minor known issue" is a strict improvement. Flag the issue, file a follow-up, merge the fix.