Keyboard shortcuts

Press or to navigate between chapters

Press S or / to search in the book

Press ? to show this help

Press Esc to hide this help

submit

jjpr submit is the manual control point. It pushes bookmarks and creates or updates PRs on the forge. Use it when you don’t want the watch loop, or when you’re debugging.

jjpr submit                           # push and update everything in the stack
jjpr submit --reviewer alice,bob      # request reviewers on every PR
jjpr submit --draft                   # create new PRs as drafts
jjpr submit --ready                   # mark existing draft PRs as ready
jjpr submit --base coworker-feat      # override auto-detected base branch

What it does

  1. Pushes all bookmarks in the stack to the remote.
  2. Creates PRs for bookmarks that don’t have one yet.
  3. Updates PR base branches to maintain the stack structure.
  4. Updates PR bodies when commit descriptions have changed.
  5. Adds or updates the stack-awareness comment on multi-PR stacks. Single-PR stacks don’t get a comment, so they look like a normal PR to reviewers.

Submit is idempotent. Run it as often as you want. After rebasing, editing commit messages, or restacking with jj rebase, re-run jjpr submit. It pushes the new commits, fixes PR base branches, and syncs descriptions. If everything is already up to date, it reports “Stack is up to date.”

When no bookmark is specified, jjpr infers the target from the working copy’s position. It finds which stack overlaps trunk()..@ and submits up to the topmost bookmark.

Flags

FlagEffect
--reviewer <users>Comma-separated list of reviewers to request
--draftCreate new PRs as drafts
--readyMark existing draft PRs as ready
--base <branch>Override auto-detected base branch
--remote <name>Override the git remote name
--no-fetchSkip git fetch before starting
--dry-runPrint what would happen without doing it

PR titles and bodies

Title and body come from the first commit’s description in each bookmark’s segment.

The body is wrapped in HTML comment markers. When you re-submit after changing a commit message, only the managed section is updated. Text you add above or below the markers (screenshots, notes, test plans) is preserved.

If you remove the markers from the PR body, jjpr stops updating that PR’s description.

The PR title is not automatically updated after creation. If you change the commit’s first line, jjpr warns you about the drift.

Drafts

--draft creates new PRs as drafts. Existing PRs are unaffected.

--ready converts every draft PR in the stack to ready-for-review.

The two flags are mutually exclusive.

Reviewers

--reviewer alice,bob requests reviewers on every PR in the stack, new and existing. Idempotent: a reviewer who’s already on a PR isn’t re-requested.

Stacking on someone else’s branch

If a commit in your stack’s ancestry has a remote bookmark that isn’t one of your own, jjpr treats it as a foreign base and targets your first PR at that branch instead of the default branch (e.g., main). The status output reflects this:

  auth (1 change, #42 open, synced)
  profile (1 change, needs push)
  (based on coworker-feat)

Use --base <branch> to override auto-detection. Useful when the coworker hasn’t pushed yet.

Conflict check

Before pushing, jjpr checks for unresolved conflicts in the stack. Conflicts halt the operation:

Error: cannot push — some commits have unresolved conflicts:

  pnnmmvmu (feat/deferment-roles): add Billings::DueDatePolicy specs

To resolve: jj edit pnnmmvmu, fix the conflicts, then re-run jjpr submit.

Stack-awareness comment

Multi-PR stacks get a comment on every PR linking the others:

This PR is part of a stack:

1. [`feat/auth`](https://github.com/o/r/pull/41)
1. **`feat/profile` <-- this PR**
1. [`feat/settings`](https://github.com/o/r/pull/43)

The list always reflects the current local stack in base-to-top order. As you rebase, split, or reorder commits, the order is recomputed every submit so each PR’s comment stays in sync with the others.

When a PR in the stack is merged or closed and its bookmark is no longer in the local graph, it moves into a collapsible history block at the bottom of the comment, rendered with strikethrough:

1. **`feat/profile` <-- this PR**
1. [`feat/settings`](https://github.com/o/r/pull/43)

<details><summary>2 earlier closed/merged PRs</summary>

1. ~~[`feat/foundation`](https://github.com/o/r/pull/39)~~
1. ~~[`feat/migration`](https://github.com/o/r/pull/40)~~

</details>

Up to seven historical entries are shown, sorted most-recent first by the forge’s merge timestamp. Older entries past the cap stay in the embedded data (so future submits can reconstruct full history) but aren’t displayed:

<summary>7 earlier closed/merged PRs (+3 older entries hidden)</summary>

The comment also embeds a base64-encoded JSON payload of the stack state inside an HTML comment marker. jjpr reads this on the next submit to inherit fossil metadata (PR numbers, merge timestamps) for PRs whose local bookmarks have been cleaned up. Don’t edit it; jjpr rewrites the whole comment on every submit.

Single-PR stacks don’t get a comment.