From 2c3b8d5f6ed6d20bb3496417a2acd57ae89c1704 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: Get branch heads and fetch from remote. ## Summary * [X] Get branch pre-fetch heads * [ ] Build candidates from prefetch heads * [X] Fetch from remotes * [X] Get branch post-fetch heads * [ ] Rebase ## Details * For branch heads, we grab local branches matching the pattern specified in args. * Pre-fetch heads query tx.base_repo() while post-fetch heads query tx.repo(); tx.repo() is the in memory Mutable repo which is updated after a fetch in a transaction. * We fetch from all branches from all configured remotes. --all_remotes affects if we use `git.fetch` or all repo remotes. Issue: #1039 --- cli/src/commands/git/sync.rs | 88 +++++++++++++++++++++++++++++++++++- 1 file changed, 86 insertions(+), 2 deletions(-) diff --git a/cli/src/commands/git/sync.rs b/cli/src/commands/git/sync.rs index 5e9eb8bf17..7d70ff7cae 100644 --- a/cli/src/commands/git/sync.rs +++ b/cli/src/commands/git/sync.rs @@ -13,11 +13,23 @@ // limitations under the License. use clap_complete::ArgValueCandidates; +use itertools::Itertools; +use jj_lib::backend::CommitId; +use jj_lib::commit::Commit; +use jj_lib::repo::Repo; +use jj_lib::revset::FailingSymbolResolver; +use jj_lib::revset::RevsetExpression; +use jj_lib::revset::RevsetIteratorExt; use jj_lib::str_util::StringPattern; use crate::cli_util::CommandHelper; +use crate::cli_util::WorkspaceCommandTransaction; use crate::command_error::CommandError; use crate::complete; +use crate::git_util::get_fetch_remotes; +use crate::git_util::get_git_repo; +use crate::git_util::git_fetch; +use crate::git_util::FetchArgs; use crate::ui::Ui; /// Sync the local `jj` repo to Git remote branch(es). @@ -59,17 +71,21 @@ pub struct GitSyncArgs { pub fn cmd_git_sync( ui: &mut Ui, command: &CommandHelper, - _args: &GitSyncArgs, + args: &GitSyncArgs, ) -> Result<(), CommandError> { - let _workspace_command = command.workspace_helper(ui)?; + let mut workspace_command = command.workspace_helper(ui)?; + let mut tx = workspace_command.start_transaction(); let guard = tracing::debug_span!("git.sync.pre-fetch").entered(); + let _prefetch_heads = get_branch_heads(tx.base_repo().as_ref(), &args.branch)?; drop(guard); let guard = tracing::debug_span!("git.sync.fetch").entered(); + git_fetch_all(ui, &mut tx, args.all_remotes)?; drop(guard); let guard = tracing::debug_span!("git.sync.post-fetch").entered(); + let _postfetch_heads = get_branch_heads(tx.repo(), &args.branch)?; drop(guard); let guard = tracing::debug_span!("git.sync.rebase").entered(); @@ -77,3 +93,71 @@ pub fn cmd_git_sync( Ok(()) } + +fn get_branch_heads( + repo: &dyn Repo, + branches: &[StringPattern], +) -> Result, CommandError> { + let mut commits: Vec = vec![]; + let local_branches = branches + .iter() + .flat_map(|pattern| { + repo.view() + .local_bookmarks_matching(pattern) + .map(|(name, _ref_target)| name) + .collect::>() + }) + .collect::>(); + + for branch in local_branches { + tracing::debug!("fetching heads for branch {branch}"); + let branch_commits: Vec = + RevsetExpression::bookmarks(StringPattern::exact(branch.to_string())) + .resolve_user_expression(repo, &FailingSymbolResolver)? + .evaluate(repo)? + .iter() + .commits(repo.store()) + .try_collect()?; + + commits.append( + &mut branch_commits + .iter() + .map(|commit| commit.id().clone()) + .collect::>(), + ); + tracing::debug!("..Ok"); + } + + Ok(commits) +} + +fn git_fetch_all( + ui: &mut Ui, + tx: &mut WorkspaceCommandTransaction, + use_all_remotes: bool, +) -> Result<(), CommandError> { + let git_repo = get_git_repo(tx.base_repo().store())?; + let remotes = get_fetch_remotes( + ui, + tx.settings(), + &git_repo, + &FetchArgs { + branch: &[StringPattern::everything()], + remotes: &[], + all_remotes: use_all_remotes, + }, + )?; + + tracing::debug!("fetching from remotes: {}", remotes.join(",")); + + git_fetch( + ui, + tx, + &git_repo, + &FetchArgs { + branch: &[StringPattern::everything()], + remotes: &remotes, + all_remotes: use_all_remotes, + }, + ) +}