Revsets¶
Jujutsu supports a functional language for selecting a set of revisions. Expressions in this language are called "revsets" (the idea comes from Mercurial). The language consists of symbols, operators, and functions.
Most jj commands accept a revset (or multiple). Many commands, such as
jj edit <revset> expect the revset to resolve to a single commit; it is an
error to pass a revset that resolves to more than one commit (or zero commits)
to such commands.
The words "revisions" and "commits" are used interchangeably in this document.
Hidden revisions¶
Most revsets search only the visible commits.
Other commits are only included if you explicitly mention them (e.g. by commit
ID, <name>@<remote> symbol, or at_operation() function).
If hidden commits are specified, their ancestors also become available to the
search space. They are included in all(), x.., ~x, etc., but not in
..visible_heads(), etc. For example, hidden_id | all() is equivalent to
hidden_id | ::(hidden_id | visible_heads()).
Symbols¶
The @ expression refers to the working copy commit in the current workspace.
Use <workspace name>@ to refer to the working-copy commit in another
workspace. Use <name>@<remote> to refer to a remote-tracking bookmark.
A full commit ID refers to a single commit. A unique prefix of the full commit ID can also be used. It is an error to use a non-unique prefix.
A full change ID refers to a visible commit with that change ID. A unique prefix of the full change ID can also be used. It is an error to use a non-unique prefix or a divergent change ID.
Use single or double quotes to prevent a symbol from being
interpreted as an expression. For example, "x-" is the symbol x-, not the
parents of symbol x. Taking shell quoting into account, you may need to use
something like jj log -r '"x-"'.
Priority¶
Jujutsu attempts to resolve a symbol in the following order:
- Tag name
- Bookmark name
- Git ref
- Commit ID or change ID
To override the priority, use the appropriate revset function. For
example, to resolve abc as a commit ID even if there happens to be a bookmark
by the same name, use commit_id(abc). This is particularly useful in scripts.
Operators¶
The following operators are supported. x and y below can be any revset, not
only symbols.
- x-: Parents of- x, can be empty.
- x+: Children of- x, can be empty.
- x::: Descendants of- x, including the commits in- xitself. Equivalent to- x::visible_heads()if no hidden revisions are mentioned.
- x..: Revisions that are not ancestors of- x. Equivalent to- ~::x, and- x..visible_heads()if no hidden revisions are mentioned.
- ::x: Ancestors of- x, including the commits in- xitself. Shorthand for- root()::x.
- ..x: Ancestors of- x, including the commits in- xitself, but excluding the root commit. Shorthand for- root()..x. Equivalent to- ::x ~ root().
- x::y: Descendants of- xthat are also ancestors of- y. Equivalent to- x:: & ::y. This is what- git logcalls- --ancestry-path x..y.
- x..y: Ancestors of- ythat are not also ancestors of- x. Equivalent to- ::y ~ ::x. This is what- git logcalls- x..y(i.e. the same as we call it).
- ::: All visible commits in the repo. Equivalent to- all(), and- root()::visible_heads()if no hidden revisions are mentioned.
- ..: All visible commits in the repo, but excluding the root commit. Equivalent to- ~root(), and- root()..visible_heads()if no hidden revisions are mentioned.
- ~x: Revisions that are not in- x.
- x & y: Revisions that are in both- xand- y.
- x ~ y: Revisions that are in- xbut not in- y.
- x | y: Revisions that are in either- xor- y(or both).
(listed in order of binding strengths)
You can use parentheses to control evaluation order, such as (x & y) | z or
x & (y | z).
Examples
Given this history:
o D
|\
| o C
| |
o | B
|/
o A
|
o root()
Operator x-
- D-⇒- {C,B}
- B-⇒- {A}
- A-⇒- {root()}
- root()-⇒- {}(empty set)
- none()-⇒- {}(empty set)
- (D|A)-⇒- {C,B,root()}
- (C|B)-⇒- {A}
Operator x+
- D+⇒- {}(empty set)
- B+⇒- {D}
- A+⇒- {B,C}
- root()+⇒- {A}
- none()+⇒- {}(empty set)
- (C|B)+⇒- {D}
- (B|root())+⇒- {D,A}
Operator x::
- D::⇒- {D}
- B::⇒- {D,B}
- A::⇒- {D,C,B,A}
- root()::⇒- {D,C,B,A,root()}
- none()::⇒- {}(empty set)
- (C|B)::⇒- {D,C,B}
Operator x..
- D..⇒- {}(empty set)
- B..⇒- {D,C}(note that, unlike- B::, this includes- C)
- A..⇒- {D,C,B}
- root()..⇒- {D,C,B,A}
- none()..⇒- {D,C,B,A,root()}
- (C|B)..⇒- {D}
Operator ::x
- ::D⇒- {D,C,B,A,root()}
- ::B⇒- {B,A,root()}
- ::A⇒- {A,root()}
- ::root()⇒- {root()}
- ::none()⇒- {}(empty set)
- ::(C|B)⇒- {C,B,A,root()}
Operator ..x
- ..D⇒- {D,C,B,A}
- ..B⇒- {B,A}
- ..A⇒- {A}
- ..root()⇒- {}(empty set)
- ..none()⇒- {}(empty set)
- ..(C|B)⇒- {C,B,A}
Operator x::y
- D::D⇒- {D}
- B::D⇒- {D,B}(note that, unlike- B..D, this includes- Band excludes- C)
- B::C⇒- {}(empty set) (note that, unlike- B..C, this excludes- C)
- A::D⇒- {D,C,B,A}
- root()::D⇒- {D,C,B,A,root()}
- none()::D⇒- {}(empty set)
- D::B⇒- {}(empty set)
- (C|B)::(C|B)⇒- {C,B}
Operator x..y
- D..D⇒- {}(empty set)
- B..D⇒- {D,C}(note that, unlike- B::D, this includes- Cand excludes- B)
- B..C⇒- {C}(note that, unlike- B::C, this includes- C)
- A..D⇒- {D,C,B}
- root()..D⇒- {D,C,B,A}
- none()..D⇒- {D,C,B,A,root()}
- D..B⇒- {}(empty set)
- (C|B)..(C|B)⇒- {}(empty set)
Functions¶
You can also specify revisions by using functions. Some functions take other revsets (expressions) as arguments.
Function argument syntax
In this documentation, optional arguments are indicated with square
brackets like [arg]. Some arguments also have an optional label which can
be used to specify that argument without specifying all previous arguments.
For instance, remote_bookmarks([bookmark_pattern], [[remote=]remote_pattern])
indicates that all of the following usages are valid:
- remote_bookmarks()
- remote_bookmarks("main")
- remote_bookmarks("main", "origin")
- remote_bookmarks("main", remote="origin")
- remote_bookmarks(remote="origin")
- 
parents(x, [depth]):parents(x)is the same asx-.parents(x, depth)returns the parents ofxat the givendepth. For instance,parents(x, 3)is equivalent tox---.
- 
children(x, [depth]):children(x)is the same asx+.children(x, depth)returns the children ofxat the givendepth. For instance,children(x, 3)is equivalent tox+++.
- 
ancestors(x, [depth]):ancestors(x)is the same as::x.ancestors(x, depth)returns the ancestors ofxlimited to the givendepth.
- 
descendants(x, [depth]):descendants(x)is the same asx::.descendants(x, depth)returns the descendants ofxlimited to the givendepth.
- 
first_parent(x, [depth]):first_parent(x)is similar toparents(x), but for merges, it only returns the first parent instead of returning all parents. Thedepthargument also works similarly, sofirst_parent(x, 2)is equivalent tofirst_parent(first_parent(x)).
- 
first_ancestors(x, [depth]): Similar toancestors(x, [depth]), but only traverses the first parent of each commit. In Git, the first parent of a merge commit is conventionally the branch into which changes are being merged, sofirst_ancestors()can be used to exclude changes made on other branches.
- 
reachable(srcs, domain): All commits reachable fromsrcswithindomain, traversing all parent and child edges.
- 
connected(x): Same asx::x. Useful whenxincludes several commits.
- 
all(): All visible commits and ancestors of commits explicitly mentioned.
- 
none(): No commits. This function is rarely useful; it is provided for completeness.
- 
change_id(prefix): Commits with the given change ID prefix. If the specified change is divergent, this resolves to multiple commits. It is an error to use a non-unique prefix. Unmatched prefix isn't an error.
- 
commit_id(prefix): Commits with the given commit ID prefix. It is an error to use a non-unique prefix. Unmatched prefix isn't an error.
- 
bookmarks([pattern]): All local bookmark targets. Ifpatternis specified, this selects the bookmarks whose name match the given string pattern. For example,bookmarks(push)would match the bookmarkspush-123andrepushedbut not the bookmarkmain. If a bookmark is in a conflicted state, all its possible targets are included.
- 
remote_bookmarks([bookmark_pattern], [[remote=]remote_pattern]): All remote bookmarks targets across all remotes. If just thebookmark_patternis specified, the bookmarks whose names match the given string pattern across all remotes are selected. If bothbookmark_patternandremote_patternare specified, the selection is further restricted to just the remotes whose names matchremote_pattern.For example, remote_bookmarks(push, ri)would match the bookmarkspush-123@originandrepushed@privatebut notpush-123@upstreamormain@originormain@upstream. If a bookmark is in a conflicted state, all its possible targets are included.While Git-tracking bookmarks can be selected by <name>@git, these bookmarks aren't included inremote_bookmarks().
- 
tracked_remote_bookmarks([bookmark_pattern], [[remote=]remote_pattern]): All targets of tracked remote bookmarks. Supports the same optional arguments asremote_bookmarks().
- 
untracked_remote_bookmarks([bookmark_pattern], [[remote=]remote_pattern]): All targets of untracked remote bookmarks. Supports the same optional arguments asremote_bookmarks().
- 
tags([pattern]): All tag targets. Ifpatternis specified, this selects the tags whose name match the given string pattern. For example,tags(v1)would match the tagsv123andrev1but not the tagv2. If a tag is in a conflicted state, all its possible targets are included.
- 
git_refs(): All Git ref targets as of the last import. If a Git ref is in a conflicted state, all its possible targets are included.
- 
git_head(): The GitHEADtarget as of the last import.
- 
visible_heads(): All visible heads (same asheads(all())if no hidden revisions are mentioned).
- 
root(): The virtual commit that is the oldest ancestor of all other commits.
- 
heads(x): Commits inxthat are not ancestors of other commits inx. Equivalent tox ~ ::x-. Note that this is different from Mercurial'sheads(x)function, which is equivalent tox ~ x-.
- 
roots(x): Commits inxthat are not descendants of other commits inx. Equivalent tox ~ x+::. Note that this is different from Mercurial'sroots(x)function, which is equivalent tox ~ x+.
- 
latest(x, [count]): Latestcountcommits inx, based on committer timestamp. The defaultcountis 1.
- 
fork_point(x): The fork point of all commits inx. The fork point is the common ancestor(s) of all commits inxwhich do not have any descendants that are also common ancestors of all commits inx. It is equivalent to the revsetheads(::x_1 & ::x_2 & ... & ::x_N), wherex_{1..N}are commits inx. Ifxresolves to a single commit,fork_point(x)resolves tox.
- 
bisect(x): Finds commits in the input set for which about half of the input set are descendants. The current implementation deals somewhat poorly with non-linear history.
- 
merges(): Merge commits.
- 
description(pattern): Commits that have a description matching the given string pattern.A non-empty description is usually terminated with newline character. For example, description(exact:"")matches commits without description, anddescription(exact:"foo\n")matches commits with description"foo\n".
- 
subject(pattern): Commits that have a subject matching the given string pattern. A subject is the first line of the description (without newline character.)
- 
author(pattern): Commits with the author's name or email matching the given string pattern. Equivalent toauthor_name(pattern) | author_email(pattern).
- 
author_name(pattern): Commits with the author's name matching the given string pattern.
- 
author_email(pattern): Commits with the author's email matching the given string pattern.
- 
author_date(pattern): Commits with author dates matching the specified date pattern.
- 
mine(): Commits where the author's email matches the email of the current user. Equivalent toauthor_email(exact-i:<user-email>)
- 
committer(pattern): Commits with the committer's name or email matching the given string pattern. Equivalent tocommitter_name(pattern) | committer_email(pattern).
- 
committer_name(pattern): Commits with the committer's name matching the given string pattern.
- 
committer_email(pattern): Commits with the committer's email matching the given string pattern.
- 
committer_date(pattern): Commits with committer dates matching the specified date pattern.
- 
signed(): Commits that are cryptographically signed.
- 
empty(): Commits modifying no files. This also includesmerges()without user modifications androot().
- 
files(expression): Commits modifying paths matching the given fileset expression.Paths are relative to the directory jjwas invoked from. A directory name will match all files in that directory and its subdirectories.For example, files(foo)will match filesfoo,foo/bar,foo/bar/baz. It will not matchfoobarorbar/foo.Some file patterns might need quoting because the expressionmust also be parsable as a revset. For example,.has to be quoted infiles(".").
- 
diff_contains(text, [files]): Commits containing diffs matching the giventextpattern line by line.The search paths can be narrowed by the filesexpression. All modified files are scanned by default, but it is likely to change in future version to respect the command line path arguments.For example, diff_contains("TODO", "src")will search revisions where "TODO" is added to or removed from files under "src".
- 
conflicts(): Commits with conflicts.
- 
present(x): Same asx, but evaluated tonone()if any of the commits inxdoesn't exist (e.g. is an unknown bookmark name.)
- 
coalesce(revsets...): Commits in the first revset in the list ofrevsetswhich does not evaluate tonone(). If all revsets evaluate tonone(), then the result ofcoalescewill also benone().
- 
working_copies(): The working copy commits across all the workspaces.
- 
at_operation(op, x): Evaluatesxat the specified operation. For example,at_operation(@-, visible_heads())will return all heads which were visible at the previous operation.Since at_operation(op, x)brings all commits that were visible at the operation to the search space,at_operation(op, x) | all()is equivalent toat_operation(op, x) | ::(at_operation(op, x | visible_heads()) | visible_heads()).
Examples
Given this history:
o E
|
| o D
|/|
| o C
| |
o | B
|/
o A
|
o root()
function reachable()
- reachable(E, A..)⇒- {E,D,C,B}
- reachable(D, A..)⇒- {E,D,C,B}
- reachable(C, A..)⇒- {E,D,C,B}
- reachable(B, A..)⇒- {E,D,C,B}
- reachable(A, A..)⇒- {}(empty set)
function connected()
- connected(E|A)⇒- {E,B,A}
- connected(D|A)⇒- {D,C,B,A}
- connected(A)⇒- {A}
function heads()
- heads(E|D)⇒- {E,D}
- heads(E|C)⇒- {E,C}
- heads(E|B)⇒- {E}
- heads(E|A)⇒- {E}
- heads(A)⇒- {A}
function roots()
- roots(E|D)⇒- {E,D}
- roots(E|C)⇒- {E,C}
- roots(E|B)⇒- {B}
- roots(E|A)⇒- {A}
- roots(A)⇒- {A}
function fork_point()
- fork_point(E|D)⇒- {B}
- fork_point(E|C)⇒- {A}
- fork_point(E|B)⇒- {B}
- fork_point(E|A)⇒- {A}
- fork_point(D|C)⇒- {C}
- fork_point(D|B)⇒- {B}
- fork_point(B|C)⇒- {A}
- fork_point(A)⇒- {A}
- fork_point(none())⇒- {}
String patterns¶
Functions that perform string matching support the following pattern syntax (the quotes are optional):
- "string"or- substring:"string": Matches strings that contain- string.
- exact:"string": Matches strings exactly equal to- string.
- glob:"pattern": Matches strings with Unix-style shell wildcard- pattern.
- regex:"pattern": Matches substrings with regular expression- pattern.
You can append -i after the kind to match case‐insensitively (e.g.
glob-i:"fix*jpeg*").
Date patterns¶
Functions that perform date matching support the following pattern syntax:
- after:"string": Matches dates exactly at or after the given date.
- before:"string": Matches dates before, but not including, the given date.
Date strings can be specified in several forms, including:
- 2024-02-01
- 2024-02-01T12:00:00
- 2024-02-01T12:00:00-08:00
- 2024-02-01 12:00:00
- 2 days ago
- 5 minutes ago
- yesterday
- yesterday 5pm
- yesterday 10:30
- yesterday 15:30
Aliases¶
New symbols and functions can be defined in the config file, by using any combination of the predefined symbols/functions and other aliases.
Alias functions can be overloaded by the number of parameters. However, builtin function will be shadowed by name, and can't co-exist with aliases.
For example:
[revset-aliases]
'HEAD' = '@-'
'user()' = 'user("me@example.org")'
'user(x)' = 'author(x) | committer(x)'
Built-in Aliases¶
The following aliases are built-in and used for certain operations. These functions are defined as aliases in order to allow you to overwrite them as needed. See revsets.toml for a comprehensive list.
- 
trunk(): Resolves to the head commit for the default bookmark of the default remote, or the remote namedupstreamororigin. This is set at the repository level upon initialization of a Jujutsu repository.If the default bookmark cannot be resolved during initialization, the default global configuration tries the bookmarks main,master, andtrunkon theupstreamandoriginremotes. If more than one potential trunk commit exists, the newest one is chosen. If none of the bookmarks exist, the revset evaluates toroot().You can override this as appropriate. If you do, make sure it always resolves to exactly one commit. For example: [revset-aliases] 'trunk()' = 'your-bookmark@your-remote'
- 
builtin_immutable_heads(): Resolves topresent(trunk()) | tags() | untracked_remote_bookmarks(). It is used as the default definition forimmutable_heads()below. It is not recommended to redefine this alias. Prefer to redefineimmutable_heads()instead.
- 
immutable_heads(): Resolves topresent(trunk()) | tags() | untracked_remote_bookmarks()by default. It is actually defined asbuiltin_immutable_heads(), and can be overridden as required. See here for details.
- 
immutable(): The set of commits thatjjtreats as immutable. This is equivalent to::(immutable_heads() | root()). It is not recommended to redefine this alias. Note that modifying this will not change whether a commit is immutable. To do that, editimmutable_heads().
- 
mutable(): The set of commits thatjjtreats as mutable. This is equivalent to~immutable(). It is not recommended to redefined this alias. Note that modifying this will not change whether a commit is immutable. To do that, editimmutable_heads().
The all: modifier¶
Certain commands (such as jj rebase) can take multiple revset arguments, and
each of these may resolve to one-or-many revisions.
If you set the ui.always-allow-large-revsets option to false, jj will not
allow revsets that resolve to more than one revision — a so-called "large
revset" — and will ask you to confirm that you want to proceed by
prefixing it with the all: modifier. This option is planned to be removed.
An all: modifier before a revset expression does not otherwise change its
meaning. Strictly speaking, it is not part of the revset language. The notation
is similar to the modifiers like glob: allowed before string
patterns.
For example, jj rebase -r w -d xyz+ will rebase w on top of the child of
xyz as long as xyz has exactly one child.
If xyz has more than one child, the all: modifier is not specified, and
ui.always-allow-large-revsets is false, jj rebase -r w -d xyz+ will return
an error.
If ui.always-allow-large-revsets was true (the default), the above command
would act as if all: was set (see the next paragraph).
With the all: modifier, jj rebase -r w -d all:xyz+ will make w into a merge
commit if xyz has more than one child. The all: modifier confirms that the
user expected xyz to have more than one child.
A more useful example: if w is a merge commit, jj rebase -s w -d all:w- -d
xyz will add xyz to the list of w's parents.
Examples¶
Show the parent(s) of the working-copy commit (like git log -1 HEAD):
jj log -r @-
Show all ancestors of the working copy (like plain git log)
jj log -r ::@
Show commits not on any remote bookmark:
jj log -r 'remote_bookmarks()..'
Show commits not on origin (if you have other remotes like fork):
jj log -r 'remote_bookmarks(remote=origin)..'
Show the initial commits in the repo (the ones Git calls "root commits"):
jj log -r 'root()+'
Show some important commits (like git --simplify-by-decoration):
jj log -r 'tags() | bookmarks()'
Show local commits leading up to the working copy, as well as descendants of those commits:
jj log -r '(remote_bookmarks()..@)::'
Show commits authored by "martinvonz" and containing the word "reset" in the description:
jj log -r 'author(martinvonz) & description(reset)'