Using Jujutsu with Gerrit Code Review¶
JJ and Gerrit share the same mental model, which makes Gerrit feel like a
natural collaboration tool for JJ. JJ tracks a "change identity" across
rewrites, and Gerrit’s Change-Id
tracks the same logical change across patch
sets. JJ and Gerrit's Change-Id
s aren’t natively compatible yet, but they’re
philosophically aligned. jj gerrit upload
bridges the gap today by adding a
Gerrit-style Change-Id
while JJ keeps its own notion of change identity on the
client. In practice, that means small, clean commits that evolve over
time, exactly how Gerrit wants you to work.
This guide assumes a basic understanding of Git, Gerrit, and Jujutsu.
Set up a Gerrit remote¶
Jujutsu communicates with Gerrit by pushing commits to a Git remote. If you're
starting from an existing Git repository with Gerrit remotes already configured,
you can use jj git init --colocate
to start using JJ in that repo. Otherwise,
set up your Gerrit remote.
# Option 1: Start JJ in an existing Git repo with Gerrit remotes
$ jj git init --colocate
# Option 2: Add a Gerrit remote to a JJ repo
$ jj git remote add gerrit https://review.gerrithub.io/yourname/yourproject
# Option 3: Clone the repo via jj
$ jj git clone https://review.gerrithub.io/your/project
If you used option 2 You can configure default values in your repository config
by appending the below to .jj/repo/config.toml
, like so:
[gerrit]
default-remote = "gerrit" # name of the Git remote to push to
default-remote-branch = "main" # target branch in Gerrit
Basic workflow¶
jj gerrit upload
takes one or more revsets, and uploads the stack of commits
ending in them to Gerrit. Each JJ change will map to a single Gerrit change
based on the JJ change ID. This should be what you want most of the time, but if
you want to associate a JJ change with a specific change already uploaded to
Gerrit, you can copy the Change-Id footer from Gerrit to the bottom of the
commit description in JJ.
Note: Gerrit identifies and updates changes by the
Change-Id
trailer. When you re-upload a commit with the sameChange-Id
, Gerrit creates a new patch set.
Upload a single change¶
# upload the previous commit (@-) for review to main
$ jj gerrit upload -r @-
Selecting revisions (revsets)¶
jj gerrit upload
accepts one or more -r/--revisions
arguments. Each argument
may expand to multiple commits. Common patterns:
-r @-
: the commit previous to the one you're currently working on-r A..B
: commits that are ancestors of B but not of A
See the revsets guide for more information.
Preview without pushing¶
Use --dry-run
to see which commits would be modified and pushed, and where,
without changing anything or contacting the remote.
$ jj gerrit upload -r '@-' --for main --dry-run
Target branch and remote selection¶
There are a few way of specifying the target remote for your projects:
- Please run
jj config set --user gerrit.default-remote-branch <branch name>
to set your default branch across all repos - Please run
jj config set --repo gerrit.default-remote-branch <branch name>
to set your default branch for this specific repo. - Use
--remote-branch <branch name>
to override this for one specific occasion.
The remote used to push is determined as follows:
- If you have more than one origin, or the origin isn't called gerrit, run
jj config set --repo gerrit.default_remote <gerrit remote name>
to set-up a default remote. - To upload to a specific remote as a one-off thing, use
--remote <remote name>
Updating changes after review¶
To address review feedback, update your revisions, then run jj gerrit
upload
again with the same revsets. Gerrit will add new patch sets to the
existing changes instead of creating new ones.
Examples:
# Edit an earlier commit in the stack
$ jj edit xcv # position on the stack to edit
--- Apply needed edits ---
$ jj gerrit upload -r xcv
Change-Id
management¶
When uploading, jj gerrit upload
adds a Change-Id
footer based on the JJ
change id. That means that any changes made to a JJ change will become a new
patch set on the Gerrit change during the next upload.
Keep this association in mind when splitting or squashing changes. For example, when splitting a change, the portion that you want associated with the original Gerrit change should remain in the original JJ change (the first half of the split). Similarly, when squashing new changes, you typically want to squash into the change that was previously uploaded to Gerrit.
If your JJ changes no longer align with the desired mapping to Gerrit changes,
you can manually copy a Gerrit Change-Id
footer into your JJ change
description to directly assign a JJ change to an exist Gerrit change.
As an alternative to jj gerrit upload
's automatic Change-Id
mapping, you
can configure JJ to automatically add Change-Id
footers to all change
descriptions:
[templates]
commit_trailers = '''
if(
!trailers.contains_key("Change-Id"),
format_gerrit_change_id_trailer(self)
)
'''
In this case, the Gerrit change mapping is defined entirely by the Change-Id
footers. When splitting or squashing changes, be sure to keep the Change-Id
footers associated with the desired changes. Be sure not to duplicate the same
Change-Id
across different changes. Gerrit will reject pushes that contain
duplicate Change-Id
s, but if the uploads are done separately, you may
unintentionally overwrite an existing change.