Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Commands which warn "The argument XX is being interpreted as a path" (eg, squash) should decline to take action if no paths match #3334

Closed
mhammond opened this issue Mar 20, 2024 · 14 comments
Labels
🐛bug Something isn't working

Comments

@mhammond
Copy link

I just did jj squash xw and the output was:

warning: The argument "xw" is being interpreted as a path. To specify a revset, pass -r "xw" instead.
Rebased 14 descendant commits

The warning was correct - I did mean to specify -r. However, it still performed a squash.

Given xw didn't actually match any paths, jj should IMO have declined to do anything.

@PhilipMetzger PhilipMetzger added the 🐛bug Something isn't working label Mar 20, 2024
@Kintaro
Copy link
Contributor

Kintaro commented Mar 28, 2024

I don't seem to be able to replicate it.

➜ jj git init
Initialized repo in "."
➜ jj describe -m "first"
Working copy now at: mynoyzzw e7859c26 (empty) first
Parent commit      : zzzzzzzz 00000000 (empty) (no description set)
➜ echo "foo" > foo
➜ jj log
@  mynoyzzw [email protected] 2024-03-28 15:56:48.000 +09:00 5bd2957b
│  first
◉  zzzzzzzz root() 00000000
➜ jj new
Working copy now at: rmvzttzn e5e243f2 (empty) (no description set)
Parent commit      : mynoyzzw 5bd2957b first
➜ jj describe -m "second"
Working copy now at: rmvzttzn c6b7467f (empty) second
Parent commit      : mynoyzzw 5bd2957b first
➜ echo "bar" > bar
➜ jj log
@  rmvzttzn [email protected] 2024-03-28 15:57:24.000 +09:00 472cd8f0
│  second
◉  mynoyzzw [email protected] 2024-03-28 15:56:48.000 +09:00 5bd2957b
│  first
◉  zzzzzzzz root() 00000000
➜ jj squash r
warning: The argument "r" is being interpreted as a path. To specify a revset, pass -r "r" instead.
Working copy now at: rmvzttzn aac9f1e6 second
Parent commit      : mynoyzzw 8ce69cc9 first
➜ jj log
@  rmvzttzn [email protected] 2024-03-28 15:57:29.000 +09:00 aac9f1e6
│  second
◉  mynoyzzw [email protected] 2024-03-28 15:57:29.000 +09:00 8ce69cc9
│  first
◉  zzzzzzzz root() 00000000

I don't see the squash being executed.

@mhammond
Copy link
Author

Thanks, I did mean to come back to this as it occurred to me later that the squash did indeed not happen - but jj took some action here, right? While the change IDs are the same, the commit IDs are different as shown both in the log and the working copy. That implies something changed, right?

I failed to reproduce exactly what I saw, but the Rebased 14 descendant commits message in particular I found confusing - there's a clear implication it changed things. So I was confused about the new state of the repo, so re-actively did jj undo but the entire process left me a little confused about exactly what happened and what state I was now in.

@Kintaro
Copy link
Contributor

Kintaro commented Mar 29, 2024

The changes to the repo you talk about are probably due to the operations log. Squash is a mutational command, so it probably still changes some internals along the way, before it figures out the path doesn't exist and a rev was probably meant.

By looking at the code (still new to jj myself though), I see that it does build diffs etc to figure out what to squash. My guess is that the changes in git commit ids stem from there?

@martinvonz
Copy link
Member

Yes, that's correct. The commits get rewritten and the timestamp changes, but the content stays the same.

@Kintaro
Copy link
Contributor

Kintaro commented Mar 29, 2024

In that case, I'd vote for closing the issue. The only improvements I see are:

  • discard the transaction when no diff to be found, but that might contradict the model
  • the warning only triggers if it's a single path, so we might improve it by checking every path if it actually exists. And it only triggers if the diff is empty at the moment. If one were to out in 2 paths, the warning doesn't trigger and it still doesn't really do anything. So the best improvement I'd see here is more of a guidance warning.

The latter case should be revisited. If more than one path is given it is clear that revisions are not intended, but there might be an argument for if a given path doesn't exist, that a warning should be given because the intention of the squash wasn't there.

@martinvonz
Copy link
Member

I think this is similar to #3367, which made jj rebase a no-op if the commit to rebase was already in place. I think it's similar in that the old behavior was more consistent but probably not what the user wanted. That's what this request is about as well - making jj squash behave differently when squashing an empty set of paths.

It's very tempting to say that we should not have these special behaviors and instead print a warning and let the user undo if they didn't want to make the change. That's what we currently do. I think @arxanas was arguing for keeping that behavior. In addition to being simpler, it also also means we don't have to worry about finding an override if the user actually did want to make the no-op change.

By the way, we have had special handling in jj describe for a very long time to make it skip rewriting the commit if the message was unchanged.

If we're going to add more of these special cases, maybe we can at least find some generic way of implementing it, so it's less code and more consistent behavior.

@Kintaro
Copy link
Contributor

Kintaro commented Mar 30, 2024

I guess the transaction could be aborted if the diff is empty.

@martinvonz
Copy link
Member

I guess the transaction could be aborted if the diff is empty.

That would not have been enough for the behavior we went with in the rebase case. That command avoids rewriting individual commits if only their timestamps would change (basically); there can still be other commits (e.g. when using multiple -s or when rebasing a merge commit) that get rewritten.

@mhammond
Copy link
Author

From my perspective, this issue is about user surprise rather than the broader issue of "rebases should be noops in some cases" - it's about helping to grow user confidence in the tool as they start using it.

The specific warning appears to only be emitted for log, move and squash - jj move doesn't really have the same scope for "unexpected rewrite" because the command-line options are such that there's less scope for ambiguity. I think there would be an expectation of log being a read-only operation, so there's no justification for that command taking special action here.

So I'd be "happy" (which isn't to say I'd be "unhappy" with another outcome) if just squash had special handling for this case, which would (IIUC) leave rebase free to do what it does now. I do understand that the implementation is such that this might be tricky to actually do though.

Naively though, I made a hacky change so an error was returned after printing the warning here - which looks like it is only emitted when there are no changes to the tree? The only test which failed was this one, which appears to already have the expectation that no material changes were made, it's just expecting success rather than failure.

@mhammond
Copy link
Author

I see the tree operations have already been done, sorry for the noise.

@bnjmnt4n
Copy link
Member

bnjmnt4n commented Apr 2, 2024

I think this issue got sidetracked slightly. It seems like there are two orthogonal issues here:

  1. Specifying a <path> argument which doesn't match any path shows a warning, but only when the resulting diff is empty.
  2. Squash will cause commits to be rewritten even if its contents are the same.

For (1), I'd like to propose that any arguments interpreted as paths should cause the whole squash operation to exit early with the warning, instead of having the warning show up only if the resulting diff is empty.

We can probably open a separate issue for (2), where we can consider more general solutions as well.

@noahmayr
Copy link
Contributor

noahmayr commented May 1, 2024

For (1), I'd like to propose that any arguments interpreted as paths should cause the whole squash operation to exit early with the warning, instead of having the warning show up only if the resulting diff is empty.

I'm nut sure that's a good idea, running jj squash docs I'd expect that all changes to ./docs will be squashed into the parent commit and not that the command will be aborted

@martinvonz
Copy link
Member

I'm nut sure that's a good idea, running jj squash docs I'd expect that all changes to ./docs will be squashed into the parent commit and not that the command will be aborted

With the commit linked above, it will be a no-op if there are no changes to docs. Are you both okay with that solution?

netbsd-srcmastr pushed a commit to NetBSD/pkgsrc that referenced this issue May 5, 2024
## [0.17.0] - 2024-05-01

### Breaking changes

* The default template aliases were replaced as follows:
  * `builtin_op_log_root(op_id: OperationId)` ->
    `format_root_operation(root: Operation)`
  * `builtin_log_root(change_id: ChangeId, commit_id: CommitId)` ->
    `format_root_commit(root: Commit)`
  * `builtin_change_id_with_hidden_and_divergent_info` ->
    `format_short_change_id_with_hidden_and_divergent_info(commit: Commit)`

* The `--revision` option of `jj rebase` is renamed to `--revisions`. The short
  alias `-r` is still supported.

### New features

* The list of conflicted paths is printed whenever the working copy changes.
  This can be disabled with the `--quiet` option.

* Commit objects in templates now have a `mine() -> Boolean` method analog to
  the same function in revsets. It evaluates to true if the email of the commit
  author matches the current `user.email`.

* Commit objects in templates now have a `contained_in(revset: String) ->
  Boolean` method.

* Operation objects in templates now have a `snapshot() -> Boolean` method that
  evaluates to true if the operation was a snapshot created by a non-mutating
  command (e.g. `jj log`).

* Revsets and templates now support single-quoted raw string literals.

* A new config option `ui.always-allow-large-revsets` has been added to
  allow large revsets expressions in some commands, without the `all:` prefix.

* A new config option `ui.allow-filesets` has been added to enable ["fileset"
  expressions](docs/filesets.md). Note that filesets are currently experimental,
  but will be enabled by default in a future release.

* A new global flag `--ignore-immutable` lets you rewrite immutable commits.

* New command `jj parallelize` that rebases a set of revisions into siblings.

* `jj status` now supports filtering by paths. For example, `jj status .` will
  only list changed files that are descendants of the current directory.

* `jj prev` and `jj next` now work when the working copy revision is a merge.

* `jj squash` now accepts a `--use-destination-message/-u` option that uses the
  description of the destination for the new squashed revision and discards the
  descriptions of the source revisions.

* You can check whether Watchman fsmonitor is enabled or installed with the new
  `jj debug watchman status` command.

* `jj rebase` now accepts revsets resolving to multiple revisions with the
   `--revisions`/`-r` option.

* `jj rebase -r` now accepts `--insert-after` and `--insert-before` options to
  customize the location of the rebased revisions.

### Fixed bugs

* Revsets now support `\`-escapes in string literal.

* The builtin diff editor now allows empty files to be selected during
  `jj split`.

* Fixed a bug with `jj split` introduced in 0.16.0 that caused it to incorrectly
  rebase the children of the revision being split if they had other parents
  (i.e. if the child was a merge).

* The `snapshot.max-new-file-size` option can now handle raw integer literals,
  interpreted as a number of bytes, where previously it could only handle string
  literals. This means that `snapshot.max-new-file-size="1"` and
  `snapshot.max-new-file-size=1` are now equivalent.

* `jj squash <path>` is now a no-op if the path argument didn't match any paths
  (it used to create new commits with bumped timestamp).
  [#3334](jj-vcs/jj#3334)
@arxanas
Copy link
Contributor

arxanas commented Jun 1, 2024

I opened #3809 as my preferred solution.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
🐛bug Something isn't working
Projects
None yet
Development

No branches or pull requests

7 participants