Skip to content

Commit

Permalink
git sync: Get heads and build candidate commits.
Browse files Browse the repository at this point in the history
## 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
  • Loading branch information
essiene committed Nov 19, 2024
1 parent 5cf013a commit d28206b
Showing 1 changed file with 72 additions and 0 deletions.
72 changes: 72 additions & 0 deletions cli/src/commands/git/sync.rs
Original file line number Diff line number Diff line change
Expand Up @@ -35,14 +35,86 @@ pub fn cmd_git_sync(
args: &GitSyncArgs,
) -> Result<(), CommandError> {
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();
drop(guard);

Ok(())
}

fn get_branch_heads(
repo: &dyn Repo,
branches: &[StringPattern],
) -> Result<Vec<CommitId>, CommandError> {
let mut commits: Vec<CommitId> = vec![];
let local_branches = branches
.iter()
.flat_map(|pattern| {
repo.view()
.local_bookmarks_matching(pattern)
.map(|(name, _ref_target)| name)
.collect::<Vec<_>>()
})
.collect::<Vec<_>>();

for branch in local_branches {
tracing::debug!("fetching heads for branch {branch}");
let branch_commits: Vec<Commit> =
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::<Vec<_>>(),
);
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,
},
)
}

0 comments on commit d28206b

Please sign in to comment.