From 2e263abec373774a3b7bf39669efb69195f71a5b Mon Sep 17 00:00:00 2001 From: Mattias Granlund Date: Sat, 16 Nov 2024 15:06:22 +0000 Subject: [PATCH] Fix bug in finding upstream commits We should be using the branch upstream instead of push remote name. --- .../src/branch_manager/branch_creation.rs | 2 +- .../src/branch_upstream_integration.rs | 4 +-- crates/gitbutler-branch-actions/src/stack.rs | 2 +- crates/gitbutler-stack/src/stack.rs | 17 ++++++------ crates/gitbutler-stack/src/stack_branch.rs | 26 ++++++++++++------- 5 files changed, 29 insertions(+), 22 deletions(-) diff --git a/crates/gitbutler-branch-actions/src/branch_manager/branch_creation.rs b/crates/gitbutler-branch-actions/src/branch_manager/branch_creation.rs index 6b4d9fc9f6..74770bfbcc 100644 --- a/crates/gitbutler-branch-actions/src/branch_manager/branch_creation.rs +++ b/crates/gitbutler-branch-actions/src/branch_manager/branch_creation.rs @@ -222,7 +222,7 @@ impl BranchManager<'_> { vb_state.find_by_source_refname_where_not_in_workspace(target) { branch.upstream_head = upstream_branch.is_some().then_some(head_commit.id()); - branch.upstream = upstream_branch; + branch.upstream = upstream_branch; // Used as remote when listing commits. branch.ownership = ownership; branch.order = order; branch.selected_for_changes = selected_for_changes; diff --git a/crates/gitbutler-branch-actions/src/branch_upstream_integration.rs b/crates/gitbutler-branch-actions/src/branch_upstream_integration.rs index 0afb1ec57f..ec6bde65de 100644 --- a/crates/gitbutler-branch-actions/src/branch_upstream_integration.rs +++ b/crates/gitbutler-branch-actions/src/branch_upstream_integration.rs @@ -36,7 +36,7 @@ pub fn integrate_upstream_commits_for_series( .iter() .find(|branch| branch.name == series_name) .ok_or(anyhow!("Series not found"))?; - let upstream_reference = subject_branch.remote_reference(remote.as_str())?; + let upstream_reference = subject_branch.remote_reference(remote.as_str()); let remote_head = repo.find_reference(&upstream_reference)?.peel_to_commit()?; let series_head = subject_branch.head_oid(&ctx.to_stack_context()?, &stack)?; @@ -59,7 +59,7 @@ pub fn integrate_upstream_commits_for_series( branch_tree: stack.tree, branch_name: &subject_branch.name, remote_head: remote_head.id(), - remote_branch_name: &subject_branch.remote_reference(&remote)?, + remote_branch_name: &subject_branch.remote_reference(&remote), strategy, }; diff --git a/crates/gitbutler-branch-actions/src/stack.rs b/crates/gitbutler-branch-actions/src/stack.rs index 84648ced1c..05130d8903 100644 --- a/crates/gitbutler-branch-actions/src/stack.rs +++ b/crates/gitbutler-branch-actions/src/stack.rs @@ -281,7 +281,7 @@ fn stack_branch_to_api_branch( // anyhow::bail!("Lets pretend this is a real error"); let remote = default_target.push_remote_name(); let upstream_reference = if stack_branch.pushed(remote.as_str(), repository)? { - stack_branch.remote_reference(remote.as_str()).ok() + Some(stack_branch.remote_reference(remote.as_str())) } else { None }; diff --git a/crates/gitbutler-stack/src/stack.rs b/crates/gitbutler-stack/src/stack.rs index 520531d685..de178ede50 100644 --- a/crates/gitbutler-stack/src/stack.rs +++ b/crates/gitbutler-stack/src/stack.rs @@ -43,7 +43,8 @@ pub struct Stack { /// If set, this means this virtual branch was originally created from `Some(branch)`. /// It can be *any* branch. pub source_refname: Option, - /// The local tracking branch, holding the state of the remote. + /// Upstream tracking branch reference, added when creating a stack from a branch. + /// Used e.g. when listing commits from a fork. pub upstream: Option, // upstream_head is the last commit on we've pushed to the upstream branch #[serde(with = "gitbutler_serde::oid_opt", default)] @@ -556,8 +557,7 @@ impl Stack { .head; let remote_name = branch_state(ctx).get_default_target()?.push_remote_name(); let upstream_refname = - RemoteRefname::from_str(&reference.remote_reference(remote_name.as_str())?) - .context("Failed to parse the remote reference for branch")?; + RemoteRefname::from_str(&reference.remote_reference(remote_name.as_str()))?; Ok(PushDetails { head: commit.id(), remote_refname: upstream_refname, @@ -925,13 +925,12 @@ fn local_reference_exists(repository: &gix::Repository, name: &str) -> Result Result { - Ok(reference - .remote_reference(state.get_default_target()?.push_remote_name().as_str()) - .and_then(|reference| local_reference_exists(repository, &reference)) - .ok() - .unwrap_or(false)) + local_reference_exists( + repository, + &branch.remote_reference(state.get_default_target()?.push_remote_name().as_str()), + ) } #[cfg(test)] diff --git a/crates/gitbutler-stack/src/stack_branch.rs b/crates/gitbutler-stack/src/stack_branch.rs index af715ead70..24b5e6bd9c 100644 --- a/crates/gitbutler-stack/src/stack_branch.rs +++ b/crates/gitbutler-stack/src/stack_branch.rs @@ -82,14 +82,15 @@ impl StackBranch { Ok(head_commit) } /// Returns a fully qualified reference with the supplied remote e.g. `refs/remotes/origin/base-branch-improvements` - pub fn remote_reference(&self, remote: &str) -> Result { - Ok(format!("refs/remotes/{}/{}", remote, self.name)) + pub fn remote_reference(&self, remote: &str) -> String { + format!("refs/remotes/{}/{}", remote, self.name) } /// Returns `true` if the reference is pushed to the provided remote pub fn pushed(&self, remote: &str, repository: &git2::Repository) -> Result { - let remote_ref = self.remote_reference(remote)?; // todo: this should probably just return false - Ok(repository.find_reference(&remote_ref).is_ok()) + Ok(repository + .find_reference(&self.remote_reference(remote)) + .is_ok()) } /// Returns the commits that are part of the branch. @@ -133,13 +134,20 @@ impl StackBranch { let default_target = stack_context.target(); let mut remote_patches: Vec> = vec![]; - let remote_name = default_target.push_remote_name(); - if self.pushed(&remote_name, repository).unwrap_or_default() { - let head_commit = repository - .find_reference(&self.remote_reference(&remote_name)?)? + + // Use remote from upstream if available, otherwise default to push remote. + let remote = stack + .upstream + .clone() + .map(|ref_name| ref_name.remote().to_owned()) + .unwrap_or(default_target.push_remote_name()); + + if self.pushed(&remote, repository).unwrap_or_default() { + let upstream_head = repository + .find_reference(self.remote_reference(&remote).as_str())? .peel_to_commit()?; repository - .log(head_commit.id(), LogUntil::Commit(previous_head), false)? + .log(upstream_head.id(), LogUntil::Commit(previous_head), false)? .into_iter() .rev() .for_each(|c| {