Skip to content

Commit

Permalink
Undo commit: Files stay in the branch that owned the undone commit
Browse files Browse the repository at this point in the history
Undoing a commit would move the files to the default branch, instead of keeping them in the source branch.
Now, the files stay in the branch the commit was undone from.
  • Loading branch information
estib-vega committed Oct 29, 2024
1 parent a216cce commit 414e45d
Show file tree
Hide file tree
Showing 2 changed files with 168 additions and 8 deletions.
101 changes: 93 additions & 8 deletions crates/gitbutler-branch-actions/src/undo_commit.rs
Original file line number Diff line number Diff line change
@@ -1,8 +1,11 @@
use std::collections::HashMap;

use anyhow::{bail, Context as _, Result};
use gitbutler_command_context::CommandContext;
use gitbutler_commit::commit_ext::CommitExt as _;
use gitbutler_diff::Hunk;
use gitbutler_repo::{rebase::cherry_rebase_group, LogUntil, RepositoryExt as _};
use gitbutler_stack::{Stack, StackId};
use gitbutler_stack::{OwnershipClaim, Stack, StackId};

use crate::VirtualBranchesExt as _;

Expand All @@ -26,7 +29,14 @@ pub(crate) fn undo_commit(

let mut branch = vb_state.get_branch_in_workspace(branch_id)?;

let new_head_commit = inner_undo_commit(ctx.repository(), branch.head(), commit_oid)?;
let UndoResult {
new_head: new_head_commit,
ownership_update,
} = inner_undo_commit(ctx.repository(), branch.head(), commit_oid)?;

for ownership in ownership_update {
branch.ownership.put(ownership);
}

branch.set_stack_head(ctx, new_head_commit, None)?;

Expand All @@ -39,20 +49,54 @@ pub(crate) fn undo_commit(
Ok(branch)
}

struct UndoResult {
new_head: git2::Oid,
ownership_update: Vec<OwnershipClaim>,
}

fn inner_undo_commit(
repository: &git2::Repository,
branch_head_commit: git2::Oid,
commit_to_remove: git2::Oid,
) -> Result<git2::Oid> {
) -> Result<UndoResult> {
let commit_to_remove = repository.find_commit(commit_to_remove)?;

if commit_to_remove.is_conflicted() {
bail!("Can not undo a conflicted commit");
}
let commit_tree = commit_to_remove
.tree()
.context("failed to get commit tree")?;
let commit_to_remove_parent = commit_to_remove.parent(0)?;
let commit_parent_tree = commit_to_remove_parent
.tree()
.context("failed to get parent tree")?;

let diff = gitbutler_diff::trees(repository, &commit_parent_tree, &commit_tree, true)?;
let diff: HashMap<_, _> = gitbutler_diff::diff_files_into_hunks(diff).collect();
let ownership_update = diff
.iter()
.filter_map(|(file_path, hunks)| {
let file_path = file_path.clone();
let hunks = hunks
.iter()
.map(Into::into)
.filter(|hunk: &Hunk| hunk.start != 0 && hunk.end != 0)
.collect::<Vec<_>>();
if hunks.is_empty() {
return None;
}
Some((file_path, hunks))
})
.map(|(file_path, hunks)| OwnershipClaim { file_path, hunks })
.collect::<Vec<_>>();

// if commit is the head, just set head to the parent
if branch_head_commit == commit_to_remove.id() {
return Ok(commit_to_remove.parent(0)?.id());
return Ok(UndoResult {
new_head: commit_to_remove_parent.id(),
ownership_update,
});
};

let commits_to_rebase = repository.l(
Expand All @@ -67,20 +111,25 @@ fn inner_undo_commit(
&commits_to_rebase,
)?;

Ok(new_head)
Ok(UndoResult {
new_head,
ownership_update,
})
}

#[cfg(test)]
mod test {
#[cfg(test)]
mod inner_undo_commit {
use std::path::PathBuf;

use gitbutler_commit::commit_ext::CommitExt as _;
use gitbutler_repo::rebase::gitbutler_merge_commits;
use gitbutler_testsupport::testing_repository::{
assert_commit_tree_matches, TestingRepository,
};

use crate::undo_commit::inner_undo_commit;
use crate::undo_commit::{inner_undo_commit, UndoResult};

#[test]
fn undoing_conflicted_commit_errors() {
Expand Down Expand Up @@ -115,9 +164,27 @@ mod test {
let b = test_repository.commit_tree(Some(&a), &[("bar.txt", "bar")]);
let c = test_repository.commit_tree(Some(&b), &[("baz.txt", "baz")]);

let new_head = inner_undo_commit(&test_repository.repository, c.id(), c.id()).unwrap();
let UndoResult {
new_head,
ownership_update,
} = inner_undo_commit(&test_repository.repository, c.id(), c.id()).unwrap();

assert_eq!(new_head, b.id(), "The new head should be C's parent");
assert_eq!(
ownership_update.len(),
1,
"Should have one ownership update"
);
assert_eq!(
ownership_update[0].file_path,
PathBuf::from("baz.txt"),
"Ownership update should be for baz.txt"
);
assert_eq!(
ownership_update[0].hunks.len(),
1,
"Ownership update should have one hunk"
);
}

#[test]
Expand All @@ -136,7 +203,10 @@ mod test {
//
// As the theirs and ours both are different to the base, it ends up
// conflicted.
let new_head = inner_undo_commit(&test_repository.repository, c.id(), b.id()).unwrap();
let UndoResult {
new_head,
ownership_update,
} = inner_undo_commit(&test_repository.repository, c.id(), b.id()).unwrap();

let new_head_commit: git2::Commit =
test_repository.repository.find_commit(new_head).unwrap();
Expand All @@ -159,6 +229,21 @@ mod test {
a.id(),
"A should be C prime's parent"
);
assert_eq!(
ownership_update.len(),
1,
"Should have one ownership update"
);
assert_eq!(
ownership_update[0].file_path,
PathBuf::from("foo.txt"),
"Ownership update should be for foo.txt"
);
assert_eq!(
ownership_update[0].hunks.len(),
1,
"Ownership update should have one hunk"
);
}
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -62,3 +62,78 @@ fn undo_commit_simple() {

assert_eq!(descriptions, vec!["commit three", "commit one"]);
}

#[test]
fn undo_commit_in_non_default_branch() {
let Test {
repository,
project,
..
} = &Test::default();

gitbutler_branch_actions::set_base_branch(
project,
&"refs/remotes/origin/master".parse().unwrap(),
)
.unwrap();

let branch_id =
gitbutler_branch_actions::create_virtual_branch(project, &BranchCreateRequest::default())
.unwrap();

// create commit
fs::write(repository.path().join("file.txt"), "content").unwrap();
let _commit1_id =
gitbutler_branch_actions::create_commit(project, branch_id, "commit one", None, false)
.unwrap();

// create commit
fs::write(repository.path().join("file2.txt"), "content2").unwrap();
fs::write(repository.path().join("file3.txt"), "content3").unwrap();
let commit2_id =
gitbutler_branch_actions::create_commit(project, branch_id, "commit two", None, false)
.unwrap();

// create commit
fs::write(repository.path().join("file4.txt"), "content4").unwrap();
let _commit3_id =
gitbutler_branch_actions::create_commit(project, branch_id, "commit three", None, false)
.unwrap();

// create default branch
// this branch should not be affected by the undo
let default_branch_id = gitbutler_branch_actions::create_virtual_branch(
project,
&BranchCreateRequest {
selected_for_changes: Some(true),
..BranchCreateRequest::default()
},
)
.unwrap();

gitbutler_branch_actions::undo_commit(project, branch_id, commit2_id).unwrap();

let mut branches = gitbutler_branch_actions::list_virtual_branches(project)
.unwrap()
.0
.into_iter();

let branch = &branches.find(|b| b.id == branch_id).unwrap();
let default_branch = &branches.find(|b| b.id == default_branch_id).unwrap();

// should be two uncommitted files now (file2.txt and file3.txt)
assert_eq!(branch.files.len(), 2);
assert_eq!(branch.commits.len(), 2);
assert_eq!(branch.commits[0].files.len(), 1);
assert_eq!(branch.commits[1].files.len(), 1);
assert_eq!(default_branch.files.len(), 0);
assert_eq!(default_branch.commits.len(), 0);

let descriptions = branch
.commits
.iter()
.map(|c| c.description.clone())
.collect::<Vec<_>>();

assert_eq!(descriptions, vec!["commit three", "commit one"]);
}

0 comments on commit 414e45d

Please sign in to comment.