revsets: change parent/children operators to foo~/foo+ (#46)

This commit is contained in:
Martin von Zweigbergk 2021-12-11 11:23:29 -08:00
parent 7f61deeb21
commit 63c90c04c8
6 changed files with 51 additions and 66 deletions

View file

@ -107,7 +107,7 @@ parent.
</tr> </tr>
<tr> <tr>
<td>Edit description (commit message) of the previous change</td> <td>Edit description (commit message) of the previous change</td>
<td><code>jj describe :@</code></td> <td><code>jj describe @~</code></td>
<td><code>git commit --amend</code> (first make sure that nothing is <td><code>git commit --amend</code> (first make sure that nothing is
staged)</td> staged)</td>
</tr> </tr>

View file

@ -167,7 +167,7 @@ o 000000000000 000000000000 1970-01-01 00:00:00.000 +00:00
(The `000000000000` commit is a virtual commit that's called the "root commit". (The `000000000000` commit is a virtual commit that's called the "root commit".
It's the root commit of every repo. The `root` symbol in the revset matches it.) It's the root commit of every repo. The `root` symbol in the revset matches it.)
There are also operators for getting the parents (`:foo`), children `foo:`, There are also operators for getting the parents (`foo~`), children (`foo+`),
ancestors (`,,foo`), descendants (`foo,,`), DAG range (`foo,,bar`, like ancestors (`,,foo`), descendants (`foo,,`), DAG range (`foo,,bar`, like
`git log --ancestry-path`), range (`foo,,,bar`, like Git's `foo..bar`). There `git log --ancestry-path`), range (`foo,,,bar`, like Git's `foo..bar`). There
are also a few more functions, such as `heads(<set>)`, which filters out are also a few more functions, such as `heads(<set>)`, which filters out
@ -195,7 +195,7 @@ Now let's see how Jujutsu deals with merge conflicts. We'll start by making some
commits: commits:
```shell script ```shell script
# Check out the grandparent of the working copy # Check out the grandparent of the working copy
$ jj co ::@ $ jj co @~~
Working copy now at: 9164f1d6a011 Working copy now at: 9164f1d6a011
Added 0 files, modified 1 files, removed 0 files Added 0 files, modified 1 files, removed 0 files
$ echo a > file1; jj close -m A $ echo a > file1; jj close -m A
@ -382,7 +382,7 @@ third commit. Remember that `jj squash` moves all the changes from one commit
into its parent. `jj squash -i` moves only part of the changes into its parent. into its parent. `jj squash -i` moves only part of the changes into its parent.
Now try that: Now try that:
```shell script ```shell script
$ jj squash -i -r :@ $ jj squash -i -r @~
Rebased 1 descendant commits Rebased 1 descendant commits
Working copy now at: 4b4c714b36aa Working copy now at: 4b4c714b36aa
``` ```
@ -391,7 +391,7 @@ the right side of the diff to have the desired end state in "ABC" by removing
the "D" line. Then close Meld. If we look the diff of the second commit, we the "D" line. Then close Meld. If we look the diff of the second commit, we
now see that all three lines got capitalized: now see that all three lines got capitalized:
```shell script ```shell script
$ jj diff -r ::@ $ jj diff -r @~~
Modified regular file file: Modified regular file file:
1 1: aA 1 1: aA
2 2: bB 2 2: bB
@ -407,7 +407,7 @@ Let's try one final command for changing the contents of an exiting commit. That
command is `jj edit`, which lets you edit the contents of a commit without command is `jj edit`, which lets you edit the contents of a commit without
checking it out. checking it out.
```shell script ```shell script
$ jj edit -r ::@ $ jj edit -r @~~
Created 2423c134ea70 ABC Created 2423c134ea70 ABC
Rebased 2 descendant commits Rebased 2 descendant commits
Working copy now at: d31c52e8ca41 Working copy now at: d31c52e8ca41
@ -415,7 +415,7 @@ Added 0 files, modified 1 files, removed 0 files
``` ```
When Meld starts, edit the right side by e.g. adding something to the first When Meld starts, edit the right side by e.g. adding something to the first
line. Then close Meld. You can now inspect the rewritten commit with line. Then close Meld. You can now inspect the rewritten commit with
`jj diff -r ::@` again and you should see your addition to the first line. `jj diff -r @~~` again and you should see your addition to the first line.
Unlike `jj squash -i`, which left the content state of the commit unchanged, Unlike `jj squash -i`, which left the content state of the commit unchanged,
`jj edit` (typically) results in a different state, which means that descendant `jj edit` (typically) results in a different state, which means that descendant
commits may have conflicts. commits may have conflicts.

View file

@ -20,11 +20,11 @@ symbol = {
literal_string = { "\"" ~ (!"\"" ~ ANY)+ ~ "\"" } literal_string = { "\"" ~ (!"\"" ~ ANY)+ ~ "\"" }
whitespace = _{ " " } whitespace = _{ " " }
parents_op = { "~" }
children_op = { "+" }
// We could use the same rule name for the shared operators if we can // We could use the same rule name for the shared operators if we can
// think of a good name. // think of a good name.
parents_op = { ":" }
children_op = { ":" }
ancestors_op = { ",," } ancestors_op = { ",," }
descendants_op = { ",," } descendants_op = { ",," }
dag_range_op = { ",," } dag_range_op = { ",," }
@ -47,16 +47,14 @@ primary = {
| symbol | symbol
} }
parents_expression = { parents_op* ~ primary } neighbors_expression = { primary ~ (parents_op | children_op)* }
children_expression = { parents_expression ~ children_op* }
range_expression = { range_expression = {
children_expression ~ dag_range_op ~ children_expression neighbors_expression ~ dag_range_op ~ neighbors_expression
| children_expression ~ range_op ~ children_expression | neighbors_expression ~ range_op ~ neighbors_expression
| ancestors_op ~ children_expression | ancestors_op ~ neighbors_expression
| children_expression ~ descendants_op | neighbors_expression ~ descendants_op
| children_expression | neighbors_expression
} }
infix_expression = { infix_expression = {

View file

@ -408,17 +408,17 @@ fn parse_range_expression_rule(
match first.as_rule() { match first.as_rule() {
Rule::ancestors_op => { Rule::ancestors_op => {
return Ok( return Ok(
parse_children_expression_rule(pairs.next().unwrap().into_inner())?.ancestors(), parse_neighbors_expression_rule(pairs.next().unwrap().into_inner())?.ancestors(),
); );
} }
Rule::children_expression => { Rule::neighbors_expression => {
// Fall through // Fall through
} }
_ => { _ => {
panic!("unxpected revset range operator rule {:?}", first.as_rule()); panic!("unxpected revset range operator rule {:?}", first.as_rule());
} }
} }
let mut expression = parse_children_expression_rule(first.into_inner())?; let mut expression = parse_neighbors_expression_rule(first.into_inner())?;
if let Some(next) = pairs.next() { if let Some(next) = pairs.next() {
match next.as_rule() { match next.as_rule() {
Rule::descendants_op => { Rule::descendants_op => {
@ -426,12 +426,12 @@ fn parse_range_expression_rule(
} }
Rule::dag_range_op => { Rule::dag_range_op => {
let heads_expression = let heads_expression =
parse_children_expression_rule(pairs.next().unwrap().into_inner())?; parse_neighbors_expression_rule(pairs.next().unwrap().into_inner())?;
expression = expression.dag_range_to(&heads_expression); expression = expression.dag_range_to(&heads_expression);
} }
Rule::range_op => { Rule::range_op => {
let expression2 = let expression2 =
parse_children_expression_rule(pairs.next().unwrap().into_inner())?; parse_neighbors_expression_rule(pairs.next().unwrap().into_inner())?;
expression = expression.range(&expression2); expression = expression.range(&expression2);
} }
_ => { _ => {
@ -442,18 +442,21 @@ fn parse_range_expression_rule(
Ok(expression) Ok(expression)
} }
fn parse_children_expression_rule( fn parse_neighbors_expression_rule(
mut pairs: Pairs<Rule>, mut pairs: Pairs<Rule>,
) -> Result<Rc<RevsetExpression>, RevsetParseError> { ) -> Result<Rc<RevsetExpression>, RevsetParseError> {
let mut expression = parse_parents_expression_rule(pairs.next().unwrap().into_inner())?; let mut expression = parse_primary_rule(pairs.next().unwrap().into_inner())?;
for operator in pairs { for operator in pairs {
match operator.as_rule() { match operator.as_rule() {
Rule::parents_op => {
expression = expression.parents();
}
Rule::children_op => { Rule::children_op => {
expression = expression.children(); expression = expression.children();
} }
_ => { _ => {
panic!( panic!(
"unxpected revset children operator rule {:?}", "unxpected revset neighbors operator rule {:?}",
operator.as_rule() operator.as_rule()
); );
} }
@ -462,22 +465,6 @@ fn parse_children_expression_rule(
Ok(expression) Ok(expression)
} }
fn parse_parents_expression_rule(
mut pairs: Pairs<Rule>,
) -> Result<Rc<RevsetExpression>, RevsetParseError> {
let first = pairs.next().unwrap();
match first.as_rule() {
Rule::primary => parse_primary_rule(first.into_inner()),
Rule::parents_op => Ok(parse_parents_expression_rule(pairs)?.parents()),
_ => {
panic!(
"unxpected revset parents operator rule {:?}",
first.as_rule()
);
}
}
}
fn parse_primary_rule(mut pairs: Pairs<Rule>) -> Result<Rc<RevsetExpression>, RevsetParseError> { fn parse_primary_rule(mut pairs: Pairs<Rule>) -> Result<Rc<RevsetExpression>, RevsetParseError> {
let first = pairs.next().unwrap(); let first = pairs.next().unwrap();
match first.as_rule() { match first.as_rule() {
@ -1337,9 +1324,9 @@ mod tests {
// Parse a quoted symbol // Parse a quoted symbol
assert_eq!(parse("\"foo\""), Ok(foo_symbol.clone())); assert_eq!(parse("\"foo\""), Ok(foo_symbol.clone()));
// Parse the "parents" operator // Parse the "parents" operator
assert_eq!(parse(":@"), Ok(checkout_symbol.parents())); assert_eq!(parse("@~"), Ok(checkout_symbol.parents()));
// Parse the "children" operator // Parse the "children" operator
assert_eq!(parse("@:"), Ok(checkout_symbol.children())); assert_eq!(parse("@+"), Ok(checkout_symbol.children()));
// Parse the "ancestors" operator // Parse the "ancestors" operator
assert_eq!(parse(",,@"), Ok(checkout_symbol.ancestors())); assert_eq!(parse(",,@"), Ok(checkout_symbol.ancestors()));
// Parse the "descendants" operator // Parse the "descendants" operator
@ -1352,14 +1339,14 @@ mod tests {
assert_eq!(parse("foo | bar"), Ok(foo_symbol.union(&bar_symbol))); assert_eq!(parse("foo | bar"), Ok(foo_symbol.union(&bar_symbol)));
// Parse the "difference" operator // Parse the "difference" operator
assert_eq!(parse("foo - bar"), Ok(foo_symbol.minus(&bar_symbol))); assert_eq!(parse("foo - bar"), Ok(foo_symbol.minus(&bar_symbol)));
// Parentheses are allowed after prefix operators // Parentheses are allowed before suffix operators
assert_eq!(parse(":(@)"), Ok(checkout_symbol.parents())); assert_eq!(parse("(@)~"), Ok(checkout_symbol.parents()));
// Space is allowed around expressions // Space is allowed around expressions
assert_eq!(parse(" ,,@ "), Ok(checkout_symbol.ancestors())); assert_eq!(parse(" ,,@ "), Ok(checkout_symbol.ancestors()));
// Space is not allowed around prefix operators // Space is not allowed around prefix operators
assert_matches!(parse(" ,, @ "), Err(RevsetParseError::SyntaxError(_))); assert_matches!(parse(" ,, @ "), Err(RevsetParseError::SyntaxError(_)));
// Incomplete parse // Incomplete parse
assert_matches!(parse("foo | :"), Err(RevsetParseError::SyntaxError(_))); assert_matches!(parse("foo | ~"), Err(RevsetParseError::SyntaxError(_)));
// Space is allowed around infix operators and function arguments // Space is allowed around infix operators and function arguments
assert_eq!( assert_eq!(
parse(" description( arg1 , arg2 ) - parents( arg1 ) - heads( ) "), parse(" description( arg1 , arg2 ) - parents( arg1 ) - heads( ) "),
@ -1375,12 +1362,12 @@ mod tests {
let foo_symbol = RevsetExpression::symbol("foo".to_string()); let foo_symbol = RevsetExpression::symbol("foo".to_string());
// Parse repeated "parents" operator // Parse repeated "parents" operator
assert_eq!( assert_eq!(
parse(":::foo"), parse("foo~~~"),
Ok(foo_symbol.parents().parents().parents()) Ok(foo_symbol.parents().parents().parents())
); );
// Parse repeated "children" operator // Parse repeated "children" operator
assert_eq!( assert_eq!(
parse("foo:::"), parse("foo+++"),
Ok(foo_symbol.children().children().children()) Ok(foo_symbol.children().children().children())
); );
// Parse repeated "ancestors"/"descendants"/"dag range" operators // Parse repeated "ancestors"/"descendants"/"dag range" operators
@ -1392,9 +1379,9 @@ mod tests {
assert_matches!(parse("foo,,bar,,"), Err(RevsetParseError::SyntaxError(_))); assert_matches!(parse("foo,,bar,,"), Err(RevsetParseError::SyntaxError(_)));
// Parse combinations of "parents"/"children" operators and the range operators. // Parse combinations of "parents"/"children" operators and the range operators.
// The former bind more strongly. // The former bind more strongly.
assert_eq!(parse(":foo:"), Ok(foo_symbol.parents().children())); assert_eq!(parse("foo~+"), Ok(foo_symbol.parents().children()));
assert_eq!(parse(":foo,,"), Ok(foo_symbol.parents().descendants())); assert_eq!(parse("foo~,,"), Ok(foo_symbol.parents().descendants()));
assert_eq!(parse(",,foo:"), Ok(foo_symbol.children().ancestors())); assert_eq!(parse(",,foo+"), Ok(foo_symbol.children().ancestors()));
} }
#[test] #[test]

View file

@ -499,18 +499,18 @@ fn test_evaluate_expression_parents(use_git: bool) {
let commit5 = graph_builder.commit_with_parents(&[&commit2]); let commit5 = graph_builder.commit_with_parents(&[&commit2]);
// The root commit has no parents // The root commit has no parents
assert_eq!(resolve_commit_ids(mut_repo.as_repo_ref(), ":root"), vec![]); assert_eq!(resolve_commit_ids(mut_repo.as_repo_ref(), "root~"), vec![]);
// Can find parents of the current checkout // Can find parents of the current checkout
mut_repo.set_checkout(commit2.id().clone()); mut_repo.set_checkout(commit2.id().clone());
assert_eq!( assert_eq!(
resolve_commit_ids(mut_repo.as_repo_ref(), ":@"), resolve_commit_ids(mut_repo.as_repo_ref(), "@~"),
vec![commit1.id().clone()] vec![commit1.id().clone()]
); );
// Can find parents of a merge commit // Can find parents of a merge commit
assert_eq!( assert_eq!(
resolve_commit_ids(mut_repo.as_repo_ref(), &format!(":{}", commit4.id().hex())), resolve_commit_ids(mut_repo.as_repo_ref(), &format!("{}~", commit4.id().hex())),
vec![commit3.id().clone(), commit2.id().clone()] vec![commit3.id().clone(), commit2.id().clone()]
); );
@ -518,7 +518,7 @@ fn test_evaluate_expression_parents(use_git: bool) {
assert_eq!( assert_eq!(
resolve_commit_ids( resolve_commit_ids(
mut_repo.as_repo_ref(), mut_repo.as_repo_ref(),
&format!(":({} | {})", commit2.id().hex(), commit3.id().hex()) &format!("({} | {})~", commit2.id().hex(), commit3.id().hex())
), ),
vec![commit1.id().clone(), root_commit.id().clone()] vec![commit1.id().clone(), root_commit.id().clone()]
); );
@ -527,7 +527,7 @@ fn test_evaluate_expression_parents(use_git: bool) {
assert_eq!( assert_eq!(
resolve_commit_ids( resolve_commit_ids(
mut_repo.as_repo_ref(), mut_repo.as_repo_ref(),
&format!(":({} | {})", commit1.id().hex(), commit2.id().hex()) &format!("({} | {})~", commit1.id().hex(), commit2.id().hex())
), ),
vec![commit1.id().clone(), root_commit.id().clone()] vec![commit1.id().clone(), root_commit.id().clone()]
); );
@ -536,7 +536,7 @@ fn test_evaluate_expression_parents(use_git: bool) {
assert_eq!( assert_eq!(
resolve_commit_ids( resolve_commit_ids(
mut_repo.as_repo_ref(), mut_repo.as_repo_ref(),
&format!(":({} | {})", commit4.id().hex(), commit5.id().hex()) &format!("({} | {})~", commit4.id().hex(), commit5.id().hex())
), ),
vec![commit3.id().clone(), commit2.id().clone()] vec![commit3.id().clone(), commit2.id().clone()]
); );
@ -569,7 +569,7 @@ fn test_evaluate_expression_children(use_git: bool) {
// Can find children of the root commit // Can find children of the root commit
assert_eq!( assert_eq!(
resolve_commit_ids(mut_repo.as_repo_ref(), "root:"), resolve_commit_ids(mut_repo.as_repo_ref(), "root+"),
vec![commit1.id().clone(), checkout_id] vec![commit1.id().clone(), checkout_id]
); );
@ -578,7 +578,7 @@ fn test_evaluate_expression_children(use_git: bool) {
assert_eq!( assert_eq!(
resolve_commit_ids( resolve_commit_ids(
mut_repo.as_repo_ref(), mut_repo.as_repo_ref(),
&format!("({} | {}):", commit1.id().hex(), commit2.id().hex()) &format!("({} | {})+", commit1.id().hex(), commit2.id().hex())
), ),
vec![ vec![
commit4.id().clone(), commit4.id().clone(),
@ -591,7 +591,7 @@ fn test_evaluate_expression_children(use_git: bool) {
assert_eq!( assert_eq!(
resolve_commit_ids( resolve_commit_ids(
mut_repo.as_repo_ref(), mut_repo.as_repo_ref(),
&format!("({} | {}):", commit3.id().hex(), commit4.id().hex()) &format!("({} | {})+", commit3.id().hex(), commit4.id().hex())
), ),
vec![commit5.id().clone()] vec![commit5.id().clone()]
); );

View file

@ -410,8 +410,8 @@ impl WorkspaceCommandHelper {
// it's another symbol, then we don't. If it's more complex, then we do // it's another symbol, then we don't. If it's more complex, then we do
// (just to be safe). TODO: Maybe make this smarter. How do we generally // (just to be safe). TODO: Maybe make this smarter. How do we generally
// figure out if a revset needs to commit the working copy? For example, // figure out if a revset needs to commit the working copy? For example,
// ":@" should perhaps not result in a new working copy commit, but // "@~" should perhaps not result in a new working copy commit, but
// "::@" should. "foo::" is probably also should, since we would // "@~~" should. "foo++" is probably also should, since we would
// otherwise need to evaluate the revset and see if "foo::" includes the // otherwise need to evaluate the revset and see if "foo::" includes the
// parent of the current checkout. Other interesting cases include some kind of // parent of the current checkout. Other interesting cases include some kind of
// reference pointing to the working copy commit. If it's a // reference pointing to the working copy commit. If it's a
@ -947,7 +947,7 @@ With the `--from` and/or `--to` options, shows the difference from/to the given
"Create a new, empty change. This may be useful if you want to make some changes \ "Create a new, empty change. This may be useful if you want to make some changes \
you're unsure of on top of the working copy. If the changes turned out to useful, \ you're unsure of on top of the working copy. If the changes turned out to useful, \
you can `jj squash` them into the previous working copy. If they turned out to be \ you can `jj squash` them into the previous working copy. If they turned out to be \
unsuccessful, you can `jj abandon` them and `jj co :@` the previous working copy.", unsuccessful, you can `jj abandon` them and `jj co @~` the previous working copy.",
) )
.arg( .arg(
Arg::with_name("revision") Arg::with_name("revision")
@ -1005,7 +1005,7 @@ With the `--from` and/or `--to` options, shows the difference from/to the given
Arg::with_name("from") Arg::with_name("from")
.long("from") .long("from")
.takes_value(true) .takes_value(true)
.default_value(":@") .default_value("@~")
.help("Revision to restore from (source)"), .help("Revision to restore from (source)"),
) )
.arg( .arg(
@ -1119,7 +1119,7 @@ A A",
.multiple(true) .multiple(true)
.help("The revision to rebase onto"), .help("The revision to rebase onto"),
); );
// TODO: It seems better to default the destination to `:@`. Maybe the working // TODO: It seems better to default the destination to `@~`. Maybe the working
// copy should be rebased on top? // copy should be rebased on top?
let backout_command = SubCommand::with_name("backout") let backout_command = SubCommand::with_name("backout")
.about("Apply the reverse of a revision on top of another revision") .about("Apply the reverse of a revision on top of another revision")