Skip to content

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-Ids 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 same Change-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-Ids, but if the uploads are done separately, you may unintentionally overwrite an existing change.