From 2bec61f29d96acc4a9e0705bfe497fe6aaf8f5e4 Mon Sep 17 00:00:00 2001 From: Essien Ita Essien <34972+essiene@users.noreply.github.com> Date: Sat, 2 Nov 2024 11:04:02 +0000 Subject: [PATCH] git sync: Rebase commits onto new branch heads. ## Summary * [X] Get branch pre-fetch heads * [X] Build candidates from prefetch heads * [X] Fetch from remotes * [X] Get branch post-fetch heads * [X] Rebase * [X] Build old -> new map * [X] transform descendants ## Details * implement CommitRewritter::repo() to return an immutable reference. * transform_descendants() roots are candidates with parents in update_record. * simplify parent merge * drop newly emptied commits * update new parents from the updated heads set if the old parents are ancestors. Issue: #1039 --- cli/src/commands/git/sync.rs | 86 +++++++++++++++++++++++++++++++++--- lib/src/rewrite.rs | 5 +++ 2 files changed, 85 insertions(+), 6 deletions(-) diff --git a/cli/src/commands/git/sync.rs b/cli/src/commands/git/sync.rs index 5269dfc1ab5..8f704ae9323 100644 --- a/cli/src/commands/git/sync.rs +++ b/cli/src/commands/git/sync.rs @@ -25,6 +25,7 @@ use jj_lib::repo::Repo; use jj_lib::revset::FailingSymbolResolver; use jj_lib::revset::RevsetExpression; use jj_lib::revset::RevsetIteratorExt; +use jj_lib::rewrite::EmptyBehaviour; use jj_lib::str_util::StringPattern; use crate::cli_util::short_commit_hash; @@ -82,7 +83,7 @@ pub fn cmd_git_sync( let guard = tracing::debug_span!("git.sync.pre-fetch").entered(); let prefetch_heads = get_branch_heads(tx.base_repo().as_ref(), &args.branch)?; - let _candidates = CandidateCommit::get(tx.repo(), &prefetch_heads)?; + let candidates = CandidateCommit::get(tx.repo(), &prefetch_heads)?; drop(guard); let guard = tracing::debug_span!("git.sync.fetch").entered(); @@ -91,7 +92,7 @@ pub fn cmd_git_sync( let guard = tracing::debug_span!("git.sync.post-fetch").entered(); let postfetch_heads = get_branch_heads(tx.repo(), &args.branch)?; - let _update_record = UpdateRecord::new( + let update_record = UpdateRecord::new( &tx, &BranchHeads { prefetch: &prefetch_heads, @@ -101,6 +102,49 @@ pub fn cmd_git_sync( drop(guard); let guard = tracing::debug_span!("git.sync.rebase").entered(); + let settings = tx.settings().clone(); + let mut num_rebased = 0; + + tx.repo_mut().transform_descendants( + &settings, + update_record.get_rebase_roots(&candidates), + |mut rewriter| { + rewriter.simplify_ancestor_merge(); + let mut updated_parents: Vec = vec![]; + + let old_parents = rewriter.new_parents().iter().cloned().collect_vec(); + + let old_commit = short_commit_hash(rewriter.old_commit().id()); + for parent in &old_parents { + let old = short_commit_hash(parent); + if let Some(updated) = update_record.maybe_update_commit(rewriter.repo(), parent) { + let new = short_commit_hash(&updated); + tracing::debug!("rebase {old_commit} from {old} to {new}"); + updated_parents.push(updated.clone()); + } else { + tracing::debug!("not rebasing {old_commit} from {old}"); + updated_parents.push(parent.clone()); + } + } + + rewriter.set_new_parents(updated_parents); + + if let Some(builder) = + rewriter.rebase_with_empty_behavior(&settings, EmptyBehaviour::AbandonNewlyEmpty)? + { + builder.write()?; + num_rebased += 1; + } + + Ok(()) + }, + )?; + + tx.finish( + ui, + format!("sync completed; {num_rebased} commits rebased to new heads"), + )?; + drop(guard); Ok(()) @@ -142,6 +186,7 @@ fn get_branch_heads( Ok(commits) } + fn set_diff(lhs: &[CommitId], rhs: &[CommitId]) -> Vec { BTreeSet::from_iter(lhs.to_vec()) .difference(&BTreeSet::from_iter(rhs.to_vec())) @@ -155,7 +200,7 @@ struct BranchHeads<'a> { } struct UpdateRecord { - _old_to_new: BTreeMap, + old_to_new: BTreeMap, } impl UpdateRecord { @@ -179,9 +224,38 @@ impl UpdateRecord { tracing::debug!("rebase children of {old} to {new}"); } - UpdateRecord { - _old_to_new: old_to_new, - } + UpdateRecord { old_to_new } + } + + /// Returns commits that need to be rebased. + /// + /// The returned commits all have parents in the `old_to_new` mapping, which + /// means that the branch their parents belong to, have advanced to new + /// commits. + fn get_rebase_roots(&self, candidates: &[CandidateCommit]) -> Vec { + candidates + .iter() + .filter_map(|candidate| { + if self.old_to_new.contains_key(&candidate.parent) { + Some(candidate.child.clone()) + } else { + None + } + }) + .collect_vec() + } + + fn maybe_update_commit(&self, repo: &dyn Repo, commit: &CommitId) -> Option { + self.old_to_new + .values() + .filter_map(|new| { + if new != commit && repo.index().is_ancestor(commit, new) { + Some(new.clone()) + } else { + None + } + }) + .next() } } diff --git a/lib/src/rewrite.rs b/lib/src/rewrite.rs index f91b41ad548..57eab0f5252 100644 --- a/lib/src/rewrite.rs +++ b/lib/src/rewrite.rs @@ -155,6 +155,11 @@ impl<'repo> CommitRewriter<'repo> { self.mut_repo } + /// Returns an immutable reference to `MutableRepo`. + pub fn repo(&mut self) -> &MutableRepo { + self.mut_repo + } + /// The commit we're rewriting. pub fn old_commit(&self) -> &Commit { &self.old_commit