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

rebase: add --insert-after and --insert-before options for --revisions #3396

Merged
merged 3 commits into from
Apr 29, 2024

Conversation

bnjmnt4n
Copy link
Member

@bnjmnt4n bnjmnt4n commented Mar 29, 2024

This adds the --insert-after and --insert-before options to jj rebase, which can only be used with -r.
This allows for commits to be moved before or after any given commits.

Closes #1188.

Checklist

If applicable:

  • I have updated CHANGELOG.md
  • I have updated the documentation (README.md, docs/, demos/)
  • I have updated the config schema (cli/src/config-schema.json)
  • I have added tests to cover my changes

@bnjmnt4n bnjmnt4n changed the title rebase: add --insert-after and --insert-before options rebase: add --insert-after and --insert-before options Mar 29, 2024
@bnjmnt4n bnjmnt4n force-pushed the push-uqxvnturzsuu branch 10 times, most recently from f999ee2 to 88a2bd7 Compare March 31, 2024 19:24
@bnjmnt4n bnjmnt4n marked this pull request as ready for review March 31, 2024 19:32
@bnjmnt4n bnjmnt4n force-pushed the push-uqxvnturzsuu branch 3 times, most recently from 5fe31d1 to f59623b Compare April 2, 2024 16:49
Copy link
Contributor

@ilyagr ilyagr left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thank you for working on this!

I'm digesting the PR a bit slowly, here are a couple of first comments.

cli/src/commands/rebase.rs Outdated Show resolved Hide resolved
cli/src/commands/rebase.rs Outdated Show resolved Hide resolved
@bnjmnt4n
Copy link
Member Author

bnjmnt4n commented Apr 3, 2024

Some notes about implementation and the DescendantRebaser API:

Considering just the first step ("extracting" the commit from its current location):

My initial approach was to mark the commit as abandoned, call MutRepo::rebase_descendants, then separately rebase the given revision on top of the destination. This didn't work as expected because abandoning a commit updates branches pointing to it to point to its parent instead, which isn't desired behavior. It can also cause the working copy commit to be abandoned, which also isn't desired in this case.

There doesn't seem to be an easy way to mark a commit as "moved" out of its location in the graph, i.e. treat it as abandoned so that its children are automatically rebased onto its parents, whilst preserving its branches. This is why I presume the current behavior is to rebase its children first onto its parents, then rebase the commit itself to its new destination, to allow the children to be auto-rebased, whilst preserving the branch of the commit when it is rewritten.

Considering the second step (shifting the commit to its new destination, and rebasing its new children on top of it):

I'm also not too sure if it's better to perform the moving in a single step: the current rebase -r implementation also does the same two steps.

@bnjmnt4n bnjmnt4n force-pushed the push-uqxvnturzsuu branch from f59623b to 932e59d Compare April 3, 2024 10:17
@bnjmnt4n
Copy link
Member Author

bnjmnt4n commented Apr 3, 2024

I created a POC of handling "moved commits" into MutRepo/DescendantRebaser in #3442, but not too sure if that's the way we want to approach this.

@ilyagr
Copy link
Contributor

ilyagr commented Apr 4, 2024

Just to set expectations: I haven't looked at this today, and it might take me a few days.

Also, somebody else might burst in to review this, which is totally fine.

@martinvonz
Copy link
Member

I created a POC of handling "moved commits" into MutRepo/DescendantRebaser in #3442, but not too sure if that's the way we want to approach this.

I very recently finished unifying the different kinds of rewritten commits so they're all recorded in parent_mappings. It would be nice if we don't need another field again.

I'm hoping the interface I'm planning will be something like mut_repo.transform_descendants(roots, callback). The callback gets something like a CommitBuilder with the new parents already set. It can the update the parents, set a new root tree, or anything it needs. It's also free to update the repo if necessary, and descendants will be updated accordingly (but additional descendants will not be detected if you mark some unrelated commit rewritten).

@bnjmnt4n
Copy link
Member Author

bnjmnt4n commented Apr 4, 2024

I've slightly modified that implementation to perform "extraction" instead, which rebases children onto its parents instead of trying to move them. I'm not tied very much to that implementation either, just a quick experiment to see how much it manages to simplify rebase.

I am interested in figuring out the new API would work. Would the new API replace all the existing MutRepo::set_rewritten_commit, MutRepo::set_abandoned_commit, etc.? Would it be called for all descendants of the roots? I guess that would be useful for more complex operations, but removing the existing methods also seems like it might the existing "common" operations more complicated.

What would the best way to implement "extraction" in the new API be (i.e. jj rebase -r A -d A-)? Would it be checking each commit's parents in the callback to see if it included A, then update the list of parents to include A's parents instead?

@bnjmnt4n bnjmnt4n force-pushed the push-uqxvnturzsuu branch from 932e59d to 64b307b Compare April 10, 2024 15:01
@martinvonz
Copy link
Member

I am interested in figuring out the new API would work. Would the new API replace all the existing MutRepo::set_rewritten_commit, MutRepo::set_abandoned_commit, etc.?

No, those functions would still exist.

Would it be called for all descendants of the roots?

Yes.

I guess that would be useful for more complex operations, but removing the existing methods also seems like it might the existing "common" operations more complicated.

Yes, I plan to keep the existing functionality. This would just add new functionality on top (or underneath, if you like).

What would the best way to implement "extraction" in the new API be (i.e. jj rebase -r A -d A-)? Would it be checking each commit's parents in the callback to see if it included A, then update the list of parents to include A's parents instead?

Exactly.

Copy link
Member

@martinvonz martinvonz left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Sorry it's taken me so long to get to this PR. It's partly that I'm wondering if there's a way we can generalize the feature so we don't have to add --insert-after/-before to other places. But it's mostly just being busy and/or prioritizing poorly. I'll try to review this later today.

cli/src/commands/rebase.rs Outdated Show resolved Hide resolved
cli/src/commands/rebase.rs Show resolved Hide resolved
cli/src/commands/rebase.rs Show resolved Hide resolved
cli/src/commands/rebase.rs Outdated Show resolved Hide resolved
cli/src/commands/rebase.rs Outdated Show resolved Hide resolved
@bnjmnt4n
Copy link
Member Author

Sorry it's taken me so long to get to this PR. It's partly that I'm wondering if there's a way we can generalize the feature so we don't have to add --insert-after/-before to other places. But it's mostly just being busy and/or prioritizing poorly. I'll try to review this later today.

No worries, I would actually like to make --insert-after/-before support more complicated revsets instead of just a single revision only, but haven't had time to think thoroughly about how that would look. I'm happy to revisit after the rebasing API refactoring is done as well.

@martinvonz martinvonz mentioned this pull request Apr 17, 2024
@bnjmnt4n
Copy link
Member Author

I've updated this PR to stack the changes on top of those in #3564.

@bnjmnt4n bnjmnt4n force-pushed the push-uqxvnturzsuu branch 2 times, most recently from e4c52a5 to 1481158 Compare April 25, 2024 21:23
@bnjmnt4n bnjmnt4n marked this pull request as draft April 26, 2024 07:42
@bnjmnt4n bnjmnt4n force-pushed the push-uqxvnturzsuu branch 2 times, most recently from 496b3e1 to 920d261 Compare April 26, 2024 16:29
@bnjmnt4n bnjmnt4n changed the title rebase: add --insert-after and --insert-before options for --revision rebase: add --insert-after and --insert-before options for --revisions Apr 26, 2024
@bnjmnt4n bnjmnt4n force-pushed the push-uqxvnturzsuu branch from 920d261 to 100a8c5 Compare April 26, 2024 17:03
@bnjmnt4n bnjmnt4n marked this pull request as ready for review April 26, 2024 17:06
@bnjmnt4n
Copy link
Member Author

Updated this; I think the biggest issue I'm not sure about is the message regarding commits displayed. It can be a bit ambiguous since the same commit might be moved multiple times during the whole rebase operation.

cli/src/commands/rebase.rs Outdated Show resolved Hide resolved
@bnjmnt4n bnjmnt4n force-pushed the push-uqxvnturzsuu branch from 100a8c5 to 987f396 Compare April 26, 2024 19:43
cli/src/commands/rebase.rs Show resolved Hide resolved
cli/src/commands/rebase.rs Show resolved Hide resolved
@bnjmnt4n bnjmnt4n force-pushed the push-uqxvnturzsuu branch from 987f396 to 289e029 Compare April 27, 2024 13:38
cli/tests/test_rebase_command.rs Show resolved Hide resolved
cli/src/commands/rebase.rs Outdated Show resolved Hide resolved
cli/src/commands/rebase.rs Outdated Show resolved Hide resolved
cli/src/commands/rebase.rs Show resolved Hide resolved
cli/tests/test_rebase_command.rs Show resolved Hide resolved
cli/src/commands/rebase.rs Outdated Show resolved Hide resolved
cli/src/commands/rebase.rs Outdated Show resolved Hide resolved
cli/tests/test_rebase_command.rs Outdated Show resolved Hide resolved
@bnjmnt4n bnjmnt4n force-pushed the push-uqxvnturzsuu branch from 289e029 to 7589a21 Compare April 28, 2024 18:53
cli/src/commands/rebase.rs Outdated Show resolved Hide resolved
@bnjmnt4n bnjmnt4n force-pushed the push-uqxvnturzsuu branch 2 times, most recently from 27bf1a4 to 1eeb7b8 Compare April 29, 2024 04:18
Copy link
Member

@martinvonz martinvonz left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks again! This was clearly a lot of work.

cli/tests/test_rebase_command.rs Outdated Show resolved Hide resolved
The `move_commits` function accepts a set of target commits to shift to
a new location given by `new_parents` and `new_children`. The roots of
the target set will be reparented onto `new_parents`. `new_children`
will then be reparented onto the heads of the target set.

The commits will be rebased in reverse topological order based on the
new set of parents of each commit, which avoids the need for multiple
sets of rebase operations.
@bnjmnt4n bnjmnt4n force-pushed the push-uqxvnturzsuu branch from 1eeb7b8 to a173522 Compare April 29, 2024 05:53
@bnjmnt4n bnjmnt4n merged commit 0e2e09a into jj-vcs:main Apr 29, 2024
16 checks passed
@bnjmnt4n bnjmnt4n deleted the push-uqxvnturzsuu branch April 29, 2024 06:19
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
3 participants