mod branch; mod iterator; use std::{ collections::HashMap, io::Write, path::{Path, PathBuf}, }; #[cfg(target_family = "unix")] use std::{ fs::Permissions, os::unix::{fs::symlink, prelude::*}, }; use anyhow::{Context, Result}; use git2::TreeEntry; use gitbutler_core::{ git::{self, CommitExt}, virtual_branches::{ self, apply_branch, branch::{BranchCreateRequest, BranchOwnershipClaims}, commit, create_virtual_branch, errors::CommitError, integrate_upstream_commits, integration::verify_branch, is_remote_branch_mergeable, is_virtual_branch_mergeable, list_remote_branches, unapply_ownership, update_branch, }, }; use pretty_assertions::assert_eq; use gitbutler_testsupport::{commit_all, virtual_branches::set_test_target, Case, Suite}; #[test] fn commit_on_branch_then_change_file_then_get_status() -> Result<()> { let suite = Suite::default(); let Case { project, project_repository, .. } = &suite.new_case_with_files(HashMap::from([ (PathBuf::from("test.txt"), "line1\nline2\nline3\nline4\n"), (PathBuf::from("test2.txt"), "line5\nline6\nline7\nline8\n"), ])); set_test_target(project_repository)?; let branch1_id = create_virtual_branch(project_repository, &BranchCreateRequest::default()) .expect("failed to create virtual branch") .id; std::fs::write( Path::new(&project.path).join("test.txt"), "line0\nline1\nline2\nline3\nline4\n", )?; let (branches, _) = virtual_branches::list_virtual_branches(project_repository)?; let branch = &branches[0]; assert_eq!(branch.files.len(), 1); assert_eq!(branch.commits.len(), 0); // commit commit( project_repository, branch1_id, "test commit", None, None, false, )?; // status (no files) let (branches, _) = virtual_branches::list_virtual_branches(project_repository)?; let branch = &branches[0]; assert_eq!(branch.files.len(), 0); assert_eq!(branch.commits.len(), 1); std::fs::write( Path::new(&project.path).join("test2.txt"), "line5\nline6\nlineBLAH\nline7\nline8\n", )?; // should have just the last change now, the other line is committed let (branches, _) = virtual_branches::list_virtual_branches(project_repository)?; let branch = &branches[0]; assert_eq!(branch.files.len(), 1); assert_eq!(branch.commits.len(), 1); Ok(()) } #[test] fn track_binary_files() -> Result<()> { let suite = Suite::default(); let Case { project_repository, project, .. } = &suite.new_case(); let file_path = Path::new("test.txt"); std::fs::write( Path::new(&project.path).join(file_path), "line1\nline2\nline3\nline4\n", )?; let file_path2 = Path::new("test2.txt"); std::fs::write( Path::new(&project.path).join(file_path2), "line5\nline6\nline7\nline8\n", )?; // add a binary file let image_data: [u8; 12] = [ 255, 0, 0, // Red pixel 0, 0, 255, // Blue pixel 255, 255, 0, // Yellow pixel 0, 255, 0, // Green pixel ]; let mut file = std::fs::File::create(Path::new(&project.path).join("image.bin"))?; file.write_all(&image_data)?; commit_all(&project_repository.git_repository); set_test_target(project_repository)?; let branch1_id = create_virtual_branch(project_repository, &BranchCreateRequest::default()) .expect("failed to create virtual branch") .id; // test file change std::fs::write( Path::new(&project.path).join(file_path2), "line5\nline6\nline7\nline8\nline9\n", )?; // add a binary file let image_data: [u8; 12] = [ 255, 0, 0, // Red pixel 0, 255, 0, // Green pixel 0, 0, 255, // Blue pixel 255, 255, 0, // Yellow pixel ]; let mut file = std::fs::File::create(Path::new(&project.path).join("image.bin"))?; file.write_all(&image_data)?; let (branches, _) = virtual_branches::list_virtual_branches(project_repository)?; let branch = &branches[0]; assert_eq!(branch.files.len(), 2); let img_file = &branch .files .iter() .find(|b| b.path.as_os_str() == "image.bin") .unwrap(); assert!(img_file.binary); let img_oid_hex = "944996dd82015a616247c72b251e41661e528ae1"; assert_eq!( img_file.hunks[0].diff, img_oid_hex, "the binary file was stored in the ODB as otherwise we wouldn't have its contents. \ It cannot easily be reconstructed from the diff-lines, or we don't attempt it." ); // commit commit( project_repository, branch1_id, "test commit", None, None, false, )?; // status (no files) let (branches, _) = virtual_branches::list_virtual_branches(project_repository).unwrap(); let commit_id = &branches[0].commits[0].id; let commit_obj = project_repository.git_repository.find_commit(*commit_id)?; let tree = commit_obj.tree()?; let files = tree_to_entry_list(&project_repository.git_repository, &tree); assert_eq!(files[0].0, "image.bin"); assert_eq!( files[0].3, img_oid_hex, "our vbranch commit tree references the binary object we previously stored" ); let image_data: [u8; 12] = [ 0, 255, 0, // Green pixel 255, 0, 0, // Red pixel 255, 255, 0, // Yellow pixel 0, 0, 255, // Blue pixel ]; let mut file = std::fs::File::create(Path::new(&project.path).join("image.bin"))?; file.write_all(&image_data)?; // commit commit( project_repository, branch1_id, "test commit", None, None, false, )?; let (branches, _) = virtual_branches::list_virtual_branches(project_repository).unwrap(); let commit_id = &branches[0].commits[0].id; // get tree from commit_id let commit_obj = project_repository.git_repository.find_commit(*commit_id)?; let tree = commit_obj.tree()?; let files = tree_to_entry_list(&project_repository.git_repository, &tree); assert_eq!(files[0].0, "image.bin"); assert_eq!(files[0].3, "ea6901a04d1eed6ebf6822f4360bda9f008fa317"); Ok(()) } #[test] fn create_branch_with_ownership() -> Result<()> { let suite = Suite::default(); let Case { project, project_repository, .. } = &suite.new_case(); set_test_target(project_repository)?; let file_path = Path::new("test.txt"); std::fs::write(Path::new(&project.path).join(file_path), "line1\nline2\n").unwrap(); let branch0 = create_virtual_branch(project_repository, &BranchCreateRequest::default()) .expect("failed to create virtual branch"); virtual_branches::get_status_by_branch(project_repository, None).expect("failed to get status"); let vb_state = project_repository.project().virtual_branches(); let branch0 = vb_state.get_branch(branch0.id).unwrap(); let branch1 = create_virtual_branch( project_repository, &BranchCreateRequest { ownership: Some(branch0.ownership), ..Default::default() }, ) .expect("failed to create virtual branch"); let statuses = virtual_branches::get_status_by_branch(project_repository, None) .expect("failed to get status") .0; let files_by_branch_id = statuses .iter() .map(|(branch, files)| (branch.id, files)) .collect::>(); assert_eq!(files_by_branch_id.len(), 2); assert_eq!(files_by_branch_id[&branch0.id].len(), 0); assert_eq!(files_by_branch_id[&branch1.id].len(), 1); Ok(()) } #[test] fn create_branch_in_the_middle() -> Result<()> { let suite = Suite::default(); let Case { project_repository, .. } = &suite.new_case(); set_test_target(project_repository)?; create_virtual_branch(project_repository, &BranchCreateRequest::default()) .expect("failed to create virtual branch"); create_virtual_branch(project_repository, &BranchCreateRequest::default()) .expect("failed to create virtual branch"); create_virtual_branch( project_repository, &BranchCreateRequest { order: Some(1), ..Default::default() }, ) .expect("failed to create virtual branch"); let vb_state = project_repository.project().virtual_branches(); let mut branches = vb_state.list_branches().expect("failed to read branches"); branches.sort_by_key(|b| b.order); assert_eq!(branches.len(), 3); assert_eq!(branches[0].name, "Virtual branch"); assert_eq!(branches[1].name, "Virtual branch 2"); assert_eq!(branches[2].name, "Virtual branch 1"); Ok(()) } #[test] fn create_branch_no_arguments() -> Result<()> { let suite = Suite::default(); let Case { project_repository, .. } = &suite.new_case(); set_test_target(project_repository)?; create_virtual_branch(project_repository, &BranchCreateRequest::default()) .expect("failed to create virtual branch"); let vb_state = project_repository.project().virtual_branches(); let branches = vb_state.list_branches().expect("failed to read branches"); assert_eq!(branches.len(), 1); assert_eq!(branches[0].name, "Virtual branch"); assert!(branches[0].applied); assert_eq!(branches[0].ownership, BranchOwnershipClaims::default()); assert_eq!(branches[0].order, 0); Ok(()) } #[test] fn hunk_expantion() -> Result<()> { let suite = Suite::default(); let Case { project_repository, project, .. } = &suite.new_case(); set_test_target(project_repository)?; let file_path = Path::new("test.txt"); std::fs::write(Path::new(&project.path).join(file_path), "line1\nline2\n")?; let branch1_id = create_virtual_branch(project_repository, &BranchCreateRequest::default()) .expect("failed to create virtual branch") .id; let branch2_id = create_virtual_branch(project_repository, &BranchCreateRequest::default()) .expect("failed to create virtual branch") .id; let statuses = virtual_branches::get_status_by_branch(project_repository, None) .expect("failed to get status") .0; let files_by_branch_id = statuses .iter() .map(|(branch, files)| (branch.id, files)) .collect::>(); assert_eq!(files_by_branch_id.len(), 2); assert_eq!(files_by_branch_id[&branch1_id].len(), 1); assert_eq!(files_by_branch_id[&branch2_id].len(), 0); // even though selected branch has changed update_branch( project_repository, virtual_branches::branch::BranchUpdateRequest { id: branch1_id, order: Some(1), ..Default::default() }, )?; update_branch( project_repository, virtual_branches::branch::BranchUpdateRequest { id: branch2_id, order: Some(0), ..Default::default() }, )?; // a slightly different hunk should still go to the same branch std::fs::write( Path::new(&project.path).join(file_path), "line1\nline2\nline3\n", )?; let statuses = virtual_branches::get_status_by_branch(project_repository, None) .expect("failed to get status") .0; let files_by_branch_id = statuses .iter() .map(|(branch, files)| (branch.id, files)) .collect::>(); assert_eq!(files_by_branch_id.len(), 2); assert_eq!(files_by_branch_id[&branch1_id].len(), 1); assert_eq!(files_by_branch_id[&branch2_id].len(), 0); Ok(()) } #[test] fn get_status_files_by_branch_no_hunks_no_branches() -> Result<()> { let suite = Suite::default(); let Case { project_repository, .. } = &suite.new_case(); set_test_target(project_repository)?; let statuses = virtual_branches::get_status_by_branch(project_repository, None) .expect("failed to get status") .0; assert_eq!(statuses.len(), 0); Ok(()) } #[test] fn get_status_files_by_branch() -> Result<()> { let suite = Suite::default(); let Case { project_repository, project, .. } = &suite.new_case(); set_test_target(project_repository)?; let file_path = Path::new("test.txt"); std::fs::write(Path::new(&project.path).join(file_path), "line1\nline2\n")?; let branch1_id = create_virtual_branch(project_repository, &BranchCreateRequest::default()) .expect("failed to create virtual branch") .id; let branch2_id = create_virtual_branch(project_repository, &BranchCreateRequest::default()) .expect("failed to create virtual branch") .id; let statuses = virtual_branches::get_status_by_branch(project_repository, None) .expect("failed to get status") .0; let files_by_branch_id = statuses .iter() .map(|(branch, files)| (branch.id, files)) .collect::>(); assert_eq!(files_by_branch_id.len(), 2); assert_eq!(files_by_branch_id[&branch1_id].len(), 1); assert_eq!(files_by_branch_id[&branch2_id].len(), 0); Ok(()) } #[test] fn move_hunks_multiple_sources() -> Result<()> { let suite = Suite::default(); let Case { project_repository, project, .. } = &suite.new_case_with_files(HashMap::from([( PathBuf::from("test.txt"), "line1\nline2\nline3\nline4\nline5\nline6\nline7\nline8\nline9\nline10\nline11\nline12\n", )])); set_test_target(project_repository)?; let branch1_id = create_virtual_branch(project_repository, &BranchCreateRequest::default()) .expect("failed to create virtual branch") .id; let branch2_id = create_virtual_branch(project_repository, &BranchCreateRequest::default()) .expect("failed to create virtual branch") .id; let branch3_id = create_virtual_branch(project_repository, &BranchCreateRequest::default()) .expect("failed to create virtual branch") .id; std::fs::write( Path::new(&project.path).join("test.txt"), "line0\nline1\nline2\nline3\nline4\nline5\nline6\nline7\nline8\nline9\nline10\nline11\nline12\nline13\n", )?; let vb_state = project.virtual_branches(); let mut branch2 = vb_state.get_branch(branch2_id)?; branch2.ownership = BranchOwnershipClaims { claims: vec!["test.txt:1-5".parse()?], }; vb_state.set_branch(branch2.clone())?; let mut branch1 = vb_state.get_branch(branch1_id)?; branch1.ownership = BranchOwnershipClaims { claims: vec!["test.txt:11-15".parse()?], }; vb_state.set_branch(branch1.clone())?; let statuses = virtual_branches::get_status_by_branch(project_repository, None) .expect("failed to get status") .0; let files_by_branch_id = statuses .iter() .map(|(branch, files)| (branch.id, files)) .collect::>(); assert_eq!(files_by_branch_id.len(), 3); assert_eq!(files_by_branch_id[&branch1_id].len(), 1); // assert_eq!(files_by_branch_id[&branch1_id][0].hunks.len(), 1); assert_eq!(files_by_branch_id[&branch2_id].len(), 1); // assert_eq!(files_by_branch_id[&branch2_id][0].hunks.len(), 1); assert_eq!(files_by_branch_id[&branch3_id].len(), 0); update_branch( project_repository, virtual_branches::branch::BranchUpdateRequest { id: branch3_id, ownership: Some("test.txt:1-5,11-15".parse()?), ..Default::default() }, )?; let statuses = virtual_branches::get_status_by_branch(project_repository, None) .expect("failed to get status") .0; let files_by_branch_id = statuses .iter() .map(|(branch, files)| (branch.id, files)) .collect::>(); assert_eq!(files_by_branch_id.len(), 3); assert_eq!(files_by_branch_id[&branch1_id].len(), 0); assert_eq!(files_by_branch_id[&branch2_id].len(), 0); assert_eq!(files_by_branch_id[&branch3_id].len(), 1); assert_eq!( files_by_branch_id[&branch3_id][Path::new("test.txt")].len(), 2 ); assert_eq!( files_by_branch_id[&branch3_id][Path::new("test.txt")][0].diff_lines, "@@ -1,3 +1,4 @@\n+line0\n line1\n line2\n line3\n" ); assert_eq!( files_by_branch_id[&branch3_id][Path::new("test.txt")][1].diff_lines, "@@ -10,3 +11,4 @@ line9\n line10\n line11\n line12\n+line13\n" ); Ok(()) } #[test] fn move_hunks_partial_explicitly() -> Result<()> { let suite = Suite::default(); let Case { project_repository, project, .. } = &suite.new_case_with_files(HashMap::from([( PathBuf::from("test.txt"), "line1\nline2\nline3\nline4\nline5\nline6\nline7\nline8\nline9\nline10\nline11\nline12\nline13\n", )])); set_test_target(project_repository)?; std::fs::write( Path::new(&project.path).join("test.txt"), "line0\nline1\nline2\nline3\nline4\nline5\nline6\nline7\nline8\nline9\nline10\nline11\nline12\nline13\nline14\n", )?; let branch1_id = create_virtual_branch(project_repository, &BranchCreateRequest::default()) .expect("failed to create virtual branch") .id; let branch2_id = create_virtual_branch(project_repository, &BranchCreateRequest::default()) .expect("failed to create virtual branch") .id; let statuses = virtual_branches::get_status_by_branch(project_repository, None) .expect("failed to get status") .0; let files_by_branch_id = statuses .iter() .map(|(branch, files)| (branch.id, files)) .collect::>(); assert_eq!(files_by_branch_id.len(), 2); assert_eq!(files_by_branch_id[&branch1_id].len(), 1); // assert_eq!(files_by_branch_id[&branch1_id][0].hunks.len(), 2); assert_eq!(files_by_branch_id[&branch2_id].len(), 0); update_branch( project_repository, virtual_branches::branch::BranchUpdateRequest { id: branch2_id, ownership: Some("test.txt:1-5".parse()?), ..Default::default() }, )?; let statuses = virtual_branches::get_status_by_branch(project_repository, None) .expect("failed to get status") .0; let files_by_branch_id = statuses .iter() .map(|(branch, files)| (branch.id, files)) .collect::>(); assert_eq!(files_by_branch_id.len(), 2); assert_eq!(files_by_branch_id[&branch1_id].len(), 1); assert_eq!( files_by_branch_id[&branch1_id][Path::new("test.txt")].len(), 1 ); assert_eq!( files_by_branch_id[&branch1_id][Path::new("test.txt")][0].diff_lines, "@@ -11,3 +12,4 @@ line10\n line11\n line12\n line13\n+line14\n" ); assert_eq!(files_by_branch_id[&branch2_id].len(), 1); assert_eq!( files_by_branch_id[&branch2_id][Path::new("test.txt")].len(), 1 ); assert_eq!( files_by_branch_id[&branch2_id][Path::new("test.txt")][0].diff_lines, "@@ -1,3 +1,4 @@\n+line0\n line1\n line2\n line3\n" ); Ok(()) } #[test] fn add_new_hunk_to_the_end() -> Result<()> { let suite = Suite::default(); let Case { project_repository, project, .. } = &suite.new_case_with_files(HashMap::from([( PathBuf::from("test.txt"), "line1\nline2\nline3\nline4\nline5\nline6\nline7\nline8\nline9\nline10\nline11\nline12\nline13\nline13\nline14\n", )])); set_test_target(project_repository)?; std::fs::write( Path::new(&project.path).join("test.txt"), "line1\nline2\nline3\nline4\nline5\nline6\nline7\nline8\nline9\nline10\nline11\nline12\nline13\nline14\nline15\n", )?; create_virtual_branch(project_repository, &BranchCreateRequest::default()) .expect("failed to create virtual branch"); let statuses = virtual_branches::get_status_by_branch(project_repository, None) .expect("failed to get status") .0; assert_eq!( statuses[0].1[Path::new("test.txt")][0].diff_lines, "@@ -11,5 +11,5 @@ line10\n line11\n line12\n line13\n-line13\n line14\n+line15\n" ); std::fs::write( Path::new(&project.path).join("test.txt"), "line0\nline1\nline2\nline3\nline4\nline5\nline6\nline7\nline8\nline9\nline10\nline11\nline12\nline13\nline14\nline15\n", )?; let statuses = virtual_branches::get_status_by_branch(project_repository, None) .expect("failed to get status") .0; assert_eq!( statuses[0].1[Path::new("test.txt")][0].diff_lines, "@@ -11,5 +12,5 @@ line10\n line11\n line12\n line13\n-line13\n line14\n+line15\n" ); assert_eq!( statuses[0].1[Path::new("test.txt")][1].diff_lines, "@@ -1,3 +1,4 @@\n+line0\n line1\n line2\n line3\n" ); Ok(()) } #[test] fn commit_id_can_be_generated_or_specified() -> Result<()> { let suite = Suite::default(); let Case { project_repository, project, .. } = &suite.new_case(); let file_path = Path::new("test.txt"); std::fs::write( Path::new(&project.path).join(file_path), "line1\nline2\nline3\nline4\n", )?; commit_all(&project_repository.git_repository); // lets make sure a change id is generated let target_oid = project_repository .git_repository .head() .unwrap() .target() .unwrap(); let target = project_repository .git_repository .find_commit(target_oid) .unwrap(); let change_id = target.change_id(); // make sure we created a change-id assert!(change_id.is_some()); // ok, make another change and specify a change-id let file_path = Path::new("test.txt"); std::fs::write( Path::new(&project.path).join(file_path), "line1\nline2\nline3\nline4\nline5\n", )?; let repository = &project_repository.git_repository; let mut index = repository.index().expect("failed to get index"); index .add_all(["."], git2::IndexAddOption::DEFAULT, None) .expect("failed to add all"); index.write().expect("failed to write index"); let oid = index.write_tree().expect("failed to write tree"); let signature = gitbutler_core::git::Signature::now("test", "test@email.com").unwrap(); let head = repository.head().expect("failed to get head"); repository .commit( Some(&head.name().unwrap()), &signature, &signature, "some commit", &repository.find_tree(oid).expect("failed to find tree"), &[&repository .find_commit( repository .refname_to_id("HEAD") .expect("failed to get head"), ) .expect("failed to find commit")], Some("my-change-id"), ) .expect("failed to commit"); let target_oid = project_repository .git_repository .head() .unwrap() .target() .unwrap(); let target = project_repository .git_repository .find_commit(target_oid) .unwrap(); let change_id = target.change_id(); // the change id should be what we specified, rather than randomly generated assert_eq!(change_id, Some("my-change-id".to_string())); Ok(()) } #[test] fn merge_vbranch_upstream_clean_rebase() -> Result<()> { let suite = Suite::default(); let Case { project_repository, project, .. } = &suite.new_case(); // create a commit and set the target let file_path = Path::new("test.txt"); std::fs::write( Path::new(&project.path).join(file_path), "line1\nline2\nline3\nline4\n", )?; commit_all(&project_repository.git_repository); let target_oid = project_repository .git_repository .head() .unwrap() .target() .unwrap(); std::fs::write( Path::new(&project.path).join(file_path), "line1\nline2\nline3\nline4\nupstream\n", )?; // add a commit to the target branch it's pointing to so there is something "upstream" commit_all(&project_repository.git_repository); let last_push = project_repository .git_repository .head() .unwrap() .target() .unwrap(); // coworker adds some work std::fs::write( Path::new(&project.path).join(file_path), "line1\nline2\nline3\nline4\nupstream\ncoworker work\n", )?; commit_all(&project_repository.git_repository); let coworker_work = project_repository .git_repository .head() .unwrap() .target() .unwrap(); //update repo ref refs/remotes/origin/master to up_target oid project_repository.git_repository.reference( &"refs/remotes/origin/master".parse().unwrap(), coworker_work, true, "update target", )?; // revert to our file std::fs::write( Path::new(&project.path).join(file_path), "line1\nline2\nline3\nline4\nupstream\n", )?; set_test_target(project_repository)?; let vb_state = project.virtual_branches(); vb_state.set_default_target(virtual_branches::target::Target { branch: "refs/remotes/origin/master".parse().unwrap(), remote_url: "origin".to_string(), sha: target_oid, push_remote_name: None, })?; // add some uncommitted work let file_path2 = Path::new("test2.txt"); std::fs::write(Path::new(&project.path).join(file_path2), "file2\n")?; // Update integration commit virtual_branches::integration::update_gitbutler_integration(&vb_state, project_repository)?; let remote_branch: git::RemoteRefname = "refs/remotes/origin/master".parse().unwrap(); let mut branch = create_virtual_branch(project_repository, &BranchCreateRequest::default()) .expect("failed to create virtual branch"); branch.upstream = Some(remote_branch.clone()); branch.head = last_push; vb_state .set_branch(branch.clone()) .context("failed to write target branch after push")?; // create the branch let (branches, _) = virtual_branches::list_virtual_branches(project_repository)?; let branch1 = &branches[0]; assert_eq!(branch1.files.len(), 1); assert_eq!(branch1.commits.len(), 1); // assert_eq!(branch1.upstream.as_ref().unwrap().commits.len(), 1); integrate_upstream_commits(project_repository, branch1.id, None)?; let (branches, _) = virtual_branches::list_virtual_branches(project_repository)?; let branch1 = &branches[0]; let contents = std::fs::read(Path::new(&project.path).join(file_path))?; assert_eq!( "line1\nline2\nline3\nline4\nupstream\ncoworker work\n", String::from_utf8(contents)? ); let contents = std::fs::read(Path::new(&project.path).join(file_path2))?; assert_eq!("file2\n", String::from_utf8(contents)?); assert_eq!(branch1.files.len(), 1); assert_eq!(branch1.commits.len(), 2); // assert_eq!(branch1.upstream.as_ref().unwrap().commits.len(), 0); Ok(()) } #[tokio::test] async fn merge_vbranch_upstream_conflict() -> Result<()> { let suite = Suite::default(); let mut case = suite.new_case(); let project = &case.project; suite .projects .update(&gitbutler_core::projects::UpdateRequest { id: project.id, ok_with_force_push: Some(false), ..Default::default() }) .await .unwrap(); case = case.refresh(&suite); let project_repository = &case.project_repository; let project = &case.project; // create a commit and set the target let file_path = Path::new("test.txt"); std::fs::write( Path::new(&project.path).join(file_path), "line1\nline2\nline3\nline4\n", )?; commit_all(&project_repository.git_repository); let target_oid = project_repository .git_repository .head() .unwrap() .target() .unwrap(); std::fs::write( Path::new(&project.path).join(file_path), "line1\nline2\nline3\nline4\nupstream\n", )?; // add a commit to the target branch it's pointing to so there is something "upstream" commit_all(&project_repository.git_repository); let last_push = project_repository .git_repository .head() .unwrap() .target() .unwrap(); // coworker adds some work std::fs::write( Path::new(&project.path).join(file_path), "line1\nline2\nline3\nline4\nupstream\ncoworker work\n", )?; commit_all(&project_repository.git_repository); let coworker_work = project_repository .git_repository .head() .unwrap() .target() .unwrap(); //update repo ref refs/remotes/origin/master to up_target oid project_repository.git_repository.reference( &"refs/remotes/origin/master".parse().unwrap(), coworker_work, true, "update target", )?; // revert to our file std::fs::write( Path::new(&project.path).join(file_path), "line1\nline2\nline3\nline4\nupstream\n", )?; set_test_target(project_repository)?; let vb_state = project.virtual_branches(); vb_state.set_default_target(virtual_branches::target::Target { branch: "refs/remotes/origin/master".parse().unwrap(), remote_url: "origin".to_string(), sha: target_oid, push_remote_name: None, })?; // add some uncommitted work std::fs::write( Path::new(&project.path).join(file_path), "line1\nline2\nline3\nline4\nupstream\nother side\n", )?; let remote_branch: git::RemoteRefname = "refs/remotes/origin/master".parse().unwrap(); let mut branch = create_virtual_branch(project_repository, &BranchCreateRequest::default()) .expect("failed to create virtual branch"); branch.upstream = Some(remote_branch.clone()); branch.head = last_push; vb_state .set_branch(branch.clone()) .context("failed to write target branch after push")?; // create the branch let (branches, _) = virtual_branches::list_virtual_branches(project_repository)?; let branch1 = &branches[0]; assert_eq!(branch1.files.len(), 1); assert_eq!(branch1.commits.len(), 1); // assert_eq!(branch1.upstream.as_ref().unwrap().commits.len(), 1); integrate_upstream_commits(project_repository, branch1.id, None)?; let (branches, _) = virtual_branches::list_virtual_branches(project_repository)?; let branch1 = &branches[0]; let contents = std::fs::read(Path::new(&project.path).join(file_path))?; assert_eq!( "line1\nline2\nline3\nline4\nupstream\n<<<<<<< ours\nother side\n=======\ncoworker work\n>>>>>>> theirs\n", String::from_utf8(contents)? ); assert_eq!(branch1.files.len(), 1); assert_eq!(branch1.commits.len(), 1); assert!(branch1.conflicted); // fix the conflict std::fs::write( Path::new(&project.path).join(file_path), "line1\nline2\nline3\nline4\nupstream\nother side\ncoworker work\n", )?; // make gb see the conflict resolution let (branches, _) = virtual_branches::list_virtual_branches(project_repository)?; assert!(branches[0].conflicted); // commit the merge resolution commit( project_repository, branch1.id, "fix merge conflict", None, None, false, )?; let (branches, _) = virtual_branches::list_virtual_branches(project_repository)?; let branch1 = &branches[0]; assert!(!branch1.conflicted); assert_eq!(branch1.files.len(), 0); assert_eq!(branch1.commits.len(), 3); // make sure the last commit was a merge commit (2 parents) let last_id = &branch1.commits[0].id; let last_commit = project_repository.git_repository.find_commit(*last_id)?; assert_eq!(last_commit.parent_count(), 2); Ok(()) } #[test] fn unapply_ownership_partial() -> Result<()> { let suite = Suite::default(); let Case { project_repository, project, .. } = &suite.new_case_with_files(HashMap::from([( PathBuf::from("test.txt"), "line1\nline2\nline3\nline4\n", )])); set_test_target(project_repository)?; std::fs::write( Path::new(&project.path).join("test.txt"), "line1\nline2\nline3\nline4\nbranch1\n", )?; create_virtual_branch(project_repository, &BranchCreateRequest::default()) .expect("failed to create virtual branch"); let (branches, _) = virtual_branches::list_virtual_branches(project_repository)?; assert_eq!(branches.len(), 1); assert_eq!(branches[0].files.len(), 1); assert_eq!(branches[0].ownership.claims.len(), 1); assert_eq!(branches[0].files[0].hunks.len(), 1); assert_eq!(branches[0].ownership.claims[0].hunks.len(), 1); assert_eq!( std::fs::read_to_string(Path::new(&project.path).join("test.txt"))?, "line1\nline2\nline3\nline4\nbranch1\n" ); unapply_ownership(project_repository, &"test.txt:2-6".parse().unwrap()).unwrap(); let (branches, _) = virtual_branches::list_virtual_branches(project_repository)?; assert_eq!(branches.len(), 1); assert_eq!(branches[0].files.len(), 0); assert_eq!(branches[0].ownership.claims.len(), 0); assert_eq!( std::fs::read_to_string(Path::new(&project.path).join("test.txt"))?, "line1\nline2\nline3\nline4\n" ); Ok(()) } #[test] fn unapply_branch() -> Result<()> { let suite = Suite::default(); let Case { project, project_repository, .. } = &suite.new_case(); // create a commit and set the target let file_path = Path::new("test.txt"); std::fs::write( Path::new(&project.path).join(file_path), "line1\nline2\nline3\nline4\n", )?; commit_all(&project_repository.git_repository); set_test_target(project_repository)?; std::fs::write( Path::new(&project.path).join(file_path), "line1\nline2\nline3\nline4\nbranch1\n", )?; let file_path2 = Path::new("test2.txt"); std::fs::write(Path::new(&project.path).join(file_path2), "line5\nline6\n")?; let branch1_id = create_virtual_branch(project_repository, &BranchCreateRequest::default()) .expect("failed to create virtual branch") .id; let branch2_id = create_virtual_branch(project_repository, &BranchCreateRequest::default()) .expect("failed to create virtual branch") .id; update_branch( project_repository, virtual_branches::branch::BranchUpdateRequest { id: branch2_id, ownership: Some("test2.txt:1-3".parse()?), ..Default::default() }, )?; let contents = std::fs::read(Path::new(&project.path).join(file_path))?; assert_eq!( "line1\nline2\nline3\nline4\nbranch1\n", String::from_utf8(contents)? ); let contents = std::fs::read(Path::new(&project.path).join(file_path2))?; assert_eq!("line5\nline6\n", String::from_utf8(contents)?); let (branches, _) = virtual_branches::list_virtual_branches(project_repository)?; let branch = &branches.iter().find(|b| b.id == branch1_id).unwrap(); assert_eq!(branch.files.len(), 1); assert!(branch.active); virtual_branches::unapply_branch(project_repository, branch1_id)?; let contents = std::fs::read(Path::new(&project.path).join(file_path))?; assert_eq!("line1\nline2\nline3\nline4\n", String::from_utf8(contents)?); let contents = std::fs::read(Path::new(&project.path).join(file_path2))?; assert_eq!("line5\nline6\n", String::from_utf8(contents)?); let (branches, _) = virtual_branches::list_virtual_branches(project_repository)?; let branch = &branches.iter().find(|b| b.id == branch1_id).unwrap(); assert_eq!(branch.files.len(), 1); assert!(!branch.active); apply_branch(project_repository, branch1_id, None)?; let contents = std::fs::read(Path::new(&project.path).join(file_path))?; assert_eq!( "line1\nline2\nline3\nline4\nbranch1\n", String::from_utf8(contents)? ); let contents = std::fs::read(Path::new(&project.path).join(file_path2))?; assert_eq!("line5\nline6\n", String::from_utf8(contents)?); let (branches, _) = virtual_branches::list_virtual_branches(project_repository)?; let branch = &branches.iter().find(|b| b.id == branch1_id).unwrap(); assert_eq!(branch.files.len(), 1); assert!(branch.active); Ok(()) } #[test] fn apply_unapply_added_deleted_files() -> Result<()> { let suite = Suite::default(); let Case { project, project_repository, .. } = &suite.new_case(); // create a commit and set the target let file_path = Path::new("test.txt"); std::fs::write(Path::new(&project.path).join(file_path), "file1\n")?; let file_path2 = Path::new("test2.txt"); std::fs::write(Path::new(&project.path).join(file_path2), "file2\n")?; commit_all(&project_repository.git_repository); set_test_target(project_repository)?; // rm file_path2, add file3 std::fs::remove_file(Path::new(&project.path).join(file_path2))?; let file_path3 = Path::new("test3.txt"); std::fs::write(Path::new(&project.path).join(file_path3), "file3\n")?; let branch2_id = create_virtual_branch(project_repository, &BranchCreateRequest::default()) .expect("failed to create virtual branch") .id; let branch3_id = create_virtual_branch(project_repository, &BranchCreateRequest::default()) .expect("failed to create virtual branch") .id; update_branch( project_repository, virtual_branches::branch::BranchUpdateRequest { id: branch2_id, ownership: Some("test2.txt:0-0".parse()?), ..Default::default() }, )?; update_branch( project_repository, virtual_branches::branch::BranchUpdateRequest { id: branch3_id, ownership: Some("test3.txt:1-2".parse()?), ..Default::default() }, )?; virtual_branches::unapply_branch(project_repository, branch2_id)?; // check that file2 is back let contents = std::fs::read(Path::new(&project.path).join(file_path2))?; assert_eq!("file2\n", String::from_utf8(contents)?); virtual_branches::unapply_branch(project_repository, branch3_id)?; // check that file3 is gone assert!(!Path::new(&project.path).join(file_path3).exists()); apply_branch(project_repository, branch2_id, None)?; // check that file2 is gone assert!(!Path::new(&project.path).join(file_path2).exists()); apply_branch(project_repository, branch3_id, None)?; // check that file3 is back let contents = std::fs::read(Path::new(&project.path).join(file_path3))?; assert_eq!("file3\n", String::from_utf8(contents)?); Ok(()) } #[test] fn detect_mergeable_branch() -> Result<()> { let suite = Suite::default(); let Case { project, project_repository, .. } = &suite.new_case(); // create a commit and set the target let file_path = Path::new("test.txt"); std::fs::write( Path::new(&project.path).join(file_path), "line1\nline2\nline3\nline4\n", )?; commit_all(&project_repository.git_repository); set_test_target(project_repository)?; std::fs::write( Path::new(&project.path).join(file_path), "line1\nline2\nline3\nline4\nbranch1\n", )?; let file_path4 = Path::new("test4.txt"); std::fs::write(Path::new(&project.path).join(file_path4), "line5\nline6\n")?; let branch1_id = create_virtual_branch(project_repository, &BranchCreateRequest::default()) .expect("failed to create virtual branch") .id; let branch2_id = create_virtual_branch(project_repository, &BranchCreateRequest::default()) .expect("failed to create virtual branch") .id; update_branch( project_repository, virtual_branches::branch::BranchUpdateRequest { id: branch2_id, ownership: Some("test4.txt:1-3".parse()?), ..Default::default() }, ) .expect("failed to update branch"); // unapply both branches and create some conflicting ones virtual_branches::unapply_branch(project_repository, branch1_id)?; virtual_branches::unapply_branch(project_repository, branch2_id)?; project_repository .git_repository .set_head(&"refs/heads/master".parse().unwrap())?; project_repository .git_repository .checkout_head(Some(&mut git2::build::CheckoutBuilder::default().force()))?; // create an upstream remote conflicting commit std::fs::write( Path::new(&project.path).join(file_path), "line1\nline2\nline3\nline4\nupstream\n", )?; commit_all(&project_repository.git_repository); let up_target = project_repository .git_repository .head() .unwrap() .target() .unwrap(); project_repository.git_repository.reference( &"refs/remotes/origin/remote_branch".parse().unwrap(), up_target, true, "update target", )?; // revert content and write a mergeable branch std::fs::write( Path::new(&project.path).join(file_path), "line1\nline2\nline3\nline4\n", )?; let file_path3 = Path::new("test3.txt"); std::fs::write(Path::new(&project.path).join(file_path3), "file3\n")?; commit_all(&project_repository.git_repository); let up_target = project_repository .git_repository .head() .unwrap() .target() .unwrap(); project_repository.git_repository.reference( &"refs/remotes/origin/remote_branch2".parse().unwrap(), up_target, true, "update target", )?; // remove file_path3 std::fs::remove_file(Path::new(&project.path).join(file_path3))?; project_repository .git_repository .set_head(&"refs/heads/gitbutler/integration".parse().unwrap())?; project_repository .git_repository .checkout_head(Some(&mut git2::build::CheckoutBuilder::default().force()))?; // create branches that conflict with our earlier branches create_virtual_branch(project_repository, &BranchCreateRequest::default()) .expect("failed to create virtual branch"); let branch4_id = create_virtual_branch(project_repository, &BranchCreateRequest::default()) .expect("failed to create virtual branch") .id; // branch3 conflicts with branch1 and remote_branch std::fs::write( Path::new(&project.path).join(file_path), "line1\nline2\nline3\nline4\nbranch3\n", )?; // branch4 conflicts with branch2 let file_path2 = Path::new("test2.txt"); std::fs::write( Path::new(&project.path).join(file_path2), "line1\nline2\nline3\nline4\nbranch4\n", )?; let vb_state = project.virtual_branches(); let mut branch4 = vb_state.get_branch(branch4_id)?; branch4.ownership = BranchOwnershipClaims { claims: vec!["test2.txt:1-6".parse()?], }; vb_state.set_branch(branch4.clone())?; let (branches, _) = virtual_branches::list_virtual_branches(project_repository)?; assert_eq!(branches.len(), 4); let branch1 = &branches.iter().find(|b| b.id == branch1_id).unwrap(); assert!(!branch1.active); assert!(!is_virtual_branch_mergeable(project_repository, branch1.id).unwrap()); let branch2 = &branches.iter().find(|b| b.id == branch2_id).unwrap(); assert!(!branch2.active); assert!(is_virtual_branch_mergeable(project_repository, branch2.id).unwrap()); let remotes = list_remote_branches(project_repository).expect("failed to list remotes"); let _remote1 = &remotes .iter() .find(|b| b.name.to_string() == "refs/remotes/origin/remote_branch") .unwrap(); assert!(!is_remote_branch_mergeable( project_repository, &"refs/remotes/origin/remote_branch".parse().unwrap() ) .unwrap()); // assert_eq!(remote1.commits.len(), 1); let _remote2 = &remotes .iter() .find(|b| b.name.to_string() == "refs/remotes/origin/remote_branch2") .unwrap(); assert!(is_remote_branch_mergeable( project_repository, &"refs/remotes/origin/remote_branch2".parse().unwrap() ) .unwrap()); // assert_eq!(remote2.commits.len(), 2); Ok(()) } #[test] fn upstream_integrated_vbranch() -> Result<()> { // ok, we need a vbranch with some work and an upstream target that also includes that work, but the base is behind // plus a branch with work not in upstream so we can see that it is not included in the vbranch let suite = Suite::default(); let Case { project_repository, project, .. } = &suite.new_case_with_files(HashMap::from([ (PathBuf::from("test.txt"), "file1\n"), (PathBuf::from("test2.txt"), "file2\n"), (PathBuf::from("test3.txt"), "file3\n"), ])); let vb_state = project_repository.project().virtual_branches(); let base_commit = project_repository .git_repository .head() .unwrap() .target() .unwrap(); std::fs::write( Path::new(&project.path).join("test.txt"), "file1\nversion2\n", )?; commit_all(&project_repository.git_repository); let upstream_commit = project_repository .git_repository .head() .unwrap() .target() .unwrap(); project_repository.git_repository.reference( &"refs/remotes/origin/master".parse().unwrap(), upstream_commit, true, "update target", )?; vb_state.set_default_target(virtual_branches::target::Target { branch: "refs/remotes/origin/master".parse().unwrap(), remote_url: "http://origin.com/project".to_string(), sha: base_commit, push_remote_name: None, })?; project_repository .git_repository .remote("origin", &"http://origin.com/project".parse().unwrap())?; virtual_branches::integration::update_gitbutler_integration(&vb_state, project_repository)?; // create vbranches, one integrated, one not let branch1_id = create_virtual_branch(project_repository, &BranchCreateRequest::default()) .expect("failed to create virtual branch") .id; let branch2_id = create_virtual_branch(project_repository, &BranchCreateRequest::default()) .expect("failed to create virtual branch") .id; let branch3_id = create_virtual_branch(project_repository, &BranchCreateRequest::default()) .expect("failed to create virtual branch") .id; std::fs::write( Path::new(&project.path).join("test2.txt"), "file2\nversion2\n", )?; std::fs::write( Path::new(&project.path).join("test3.txt"), "file3\nversion2\n", )?; update_branch( project_repository, virtual_branches::branch::BranchUpdateRequest { id: branch1_id, name: Some("integrated".to_string()), ownership: Some("test.txt:1-2".parse()?), ..Default::default() }, )?; update_branch( project_repository, virtual_branches::branch::BranchUpdateRequest { id: branch2_id, name: Some("not integrated".to_string()), ownership: Some("test2.txt:1-2".parse()?), ..Default::default() }, )?; update_branch( project_repository, virtual_branches::branch::BranchUpdateRequest { id: branch3_id, name: Some("not committed".to_string()), ownership: Some("test3.txt:1-2".parse()?), ..Default::default() }, )?; // create a new virtual branch from the remote branch commit( project_repository, branch1_id, "integrated commit", None, None, false, )?; commit( project_repository, branch2_id, "non-integrated commit", None, None, false, )?; let (branches, _) = virtual_branches::list_virtual_branches(project_repository)?; let branch1 = &branches.iter().find(|b| b.id == branch1_id).unwrap(); assert!(branch1.commits.iter().any(|c| c.is_integrated)); assert_eq!(branch1.files.len(), 0); assert_eq!(branch1.commits.len(), 1); let branch2 = &branches.iter().find(|b| b.id == branch2_id).unwrap(); assert!(!branch2.commits.iter().any(|c| c.is_integrated)); assert_eq!(branch2.files.len(), 0); assert_eq!(branch2.commits.len(), 1); let branch3 = &branches.iter().find(|b| b.id == branch3_id).unwrap(); assert!(!branch3.commits.iter().any(|c| c.is_integrated)); assert_eq!(branch3.files.len(), 1); assert_eq!(branch3.commits.len(), 0); Ok(()) } #[test] fn commit_same_hunk_twice() -> Result<()> { let suite = Suite::default(); let Case { project_repository, project, .. } = &suite.new_case_with_files(HashMap::from([( PathBuf::from("test.txt"), "line1\nline2\nline3\nline4\nline5\nmiddle\nmiddle\nmiddle\nmiddle\nline6\nline7\nline8\nline9\nline10\nmiddle\nmiddle\nmiddle\nline11\nline12\n", )])); set_test_target(project_repository)?; let branch1_id = create_virtual_branch(project_repository, &BranchCreateRequest::default()) .expect("failed to create virtual branch") .id; std::fs::write( Path::new(&project.path).join("test.txt"), "line1\npatch1\nline2\nline3\nline4\nline5\nmiddle\nmiddle\nmiddle\nmiddle\nline6\nline7\nline8\nline9\nline10\nmiddle\nmiddle\nmiddle\nline11\nline12\n", )?; let (branches, _) = virtual_branches::list_virtual_branches(project_repository)?; let branch = &branches.iter().find(|b| b.id == branch1_id).unwrap(); assert_eq!(branch.files.len(), 1); assert_eq!(branch.files[0].hunks.len(), 1); assert_eq!(branch.commits.len(), 0); // commit commit( project_repository, branch1_id, "first commit to test.txt", None, None, false, )?; let (branches, _) = virtual_branches::list_virtual_branches(project_repository)?; let branch = &branches.iter().find(|b| b.id == branch1_id).unwrap(); assert_eq!(branch.files.len(), 0, "no files expected"); assert_eq!(branch.commits.len(), 1, "file should have been commited"); assert_eq!(branch.commits[0].files.len(), 1, "hunks expected"); assert_eq!( branch.commits[0].files[0].hunks.len(), 1, "one hunk should have been commited" ); // update same lines std::fs::write( Path::new(&project.path).join("test.txt"), "line1\nPATCH1\nline2\nline3\nline4\nline5\nmiddle\nmiddle\nmiddle\nmiddle\nline6\nline7\nline8\nline9\nline10\nmiddle\nmiddle\nmiddle\nline11\nline12\n", )?; let (branches, _) = virtual_branches::list_virtual_branches(project_repository)?; let branch = &branches.iter().find(|b| b.id == branch1_id).unwrap(); assert_eq!(branch.files.len(), 1, "one file should be changed"); assert_eq!(branch.commits.len(), 1, "commit is still there"); commit( project_repository, branch1_id, "second commit to test.txt", None, None, false, )?; let (branches, _) = virtual_branches::list_virtual_branches(project_repository)?; let branch = &branches.iter().find(|b| b.id == branch1_id).unwrap(); assert_eq!( branch.files.len(), 0, "all changes should have been commited" ); assert_eq!(branch.commits.len(), 2, "two commits expected"); assert_eq!(branch.commits[0].files.len(), 1); assert_eq!(branch.commits[0].files[0].hunks.len(), 1); assert_eq!(branch.commits[1].files.len(), 1); assert_eq!(branch.commits[1].files[0].hunks.len(), 1); Ok(()) } #[test] fn commit_same_file_twice() -> Result<()> { let suite = Suite::default(); let Case { project_repository, project, .. } = &suite.new_case_with_files(HashMap::from([( PathBuf::from("test.txt"), "line1\nline2\nline3\nline4\nline5\nmiddle\nmiddle\nmiddle\nmiddle\nline6\nline7\nline8\nline9\nline10\nmiddle\nmiddle\nmiddle\nline11\nline12\n", )])); set_test_target(project_repository)?; let branch1_id = create_virtual_branch(project_repository, &BranchCreateRequest::default()) .expect("failed to create virtual branch") .id; std::fs::write( Path::new(&project.path).join("test.txt"), "line1\npatch1\nline2\nline3\nline4\nline5\nmiddle\nmiddle\nmiddle\nmiddle\nline6\nline7\nline8\nline9\nline10\nmiddle\nmiddle\nmiddle\nline11\nline12\n", )?; let (branches, _) = virtual_branches::list_virtual_branches(project_repository)?; let branch = &branches.iter().find(|b| b.id == branch1_id).unwrap(); assert_eq!(branch.files.len(), 1); assert_eq!(branch.files[0].hunks.len(), 1); assert_eq!(branch.commits.len(), 0); // commit commit( project_repository, branch1_id, "first commit to test.txt", None, None, false, )?; let (branches, _) = virtual_branches::list_virtual_branches(project_repository)?; let branch = &branches.iter().find(|b| b.id == branch1_id).unwrap(); assert_eq!(branch.files.len(), 0, "no files expected"); assert_eq!(branch.commits.len(), 1, "file should have been commited"); assert_eq!(branch.commits[0].files.len(), 1, "hunks expected"); assert_eq!( branch.commits[0].files[0].hunks.len(), 1, "one hunk should have been commited" ); // add second patch std::fs::write( Path::new(&project.path).join("file.txt"), "line1\npatch1\nline2\nline3\nline4\nline5\nmiddle\nmiddle\nmiddle\nmiddle\nline6\nline7\nline8\nline9\nline10\nmiddle\nmiddle\nmiddle\npatch2\nline11\nline12\n", )?; let (branches, _) = virtual_branches::list_virtual_branches(project_repository)?; let branch = &branches.iter().find(|b| b.id == branch1_id).unwrap(); assert_eq!(branch.files.len(), 1, "one file should be changed"); assert_eq!(branch.commits.len(), 1, "commit is still there"); commit( project_repository, branch1_id, "second commit to test.txt", None, None, false, )?; let (branches, _) = virtual_branches::list_virtual_branches(project_repository)?; let branch = &branches.iter().find(|b| b.id == branch1_id).unwrap(); assert_eq!( branch.files.len(), 0, "all changes should have been commited" ); assert_eq!(branch.commits.len(), 2, "two commits expected"); assert_eq!(branch.commits[0].files.len(), 1); assert_eq!(branch.commits[0].files[0].hunks.len(), 1); assert_eq!(branch.commits[1].files.len(), 1); assert_eq!(branch.commits[1].files[0].hunks.len(), 1); Ok(()) } #[test] fn commit_partial_by_hunk() -> Result<()> { let suite = Suite::default(); let Case { project_repository, project, .. } = &suite.new_case_with_files(HashMap::from([( PathBuf::from("test.txt"), "line1\nline2\nline3\nline4\nline5\nmiddle\nmiddle\nmiddle\nmiddle\nline6\nline7\nline8\nline9\nline10\nmiddle\nmiddle\nmiddle\nline11\nline12\n", )])); set_test_target(project_repository)?; let branch1_id = create_virtual_branch(project_repository, &BranchCreateRequest::default()) .expect("failed to create virtual branch") .id; std::fs::write( Path::new(&project.path).join("test.txt"), "line1\npatch1\nline2\nline3\nline4\nline5\nmiddle\nmiddle\nmiddle\nmiddle\nline6\nline7\nline8\nline9\nline10\nmiddle\nmiddle\nmiddle\npatch2\nline11\nline12\n", )?; let (branches, _) = virtual_branches::list_virtual_branches(project_repository)?; let branch = &branches.iter().find(|b| b.id == branch1_id).unwrap(); assert_eq!(branch.files.len(), 1); assert_eq!(branch.files[0].hunks.len(), 2); assert_eq!(branch.commits.len(), 0); // commit commit( project_repository, branch1_id, "first commit to test.txt", Some(&"test.txt:1-6".parse::().unwrap()), None, false, )?; let (branches, _) = virtual_branches::list_virtual_branches(project_repository)?; let branch = &branches.iter().find(|b| b.id == branch1_id).unwrap(); assert_eq!(branch.files.len(), 1); assert_eq!(branch.files[0].hunks.len(), 1); assert_eq!(branch.commits.len(), 1); assert_eq!(branch.commits[0].files.len(), 1); assert_eq!(branch.commits[0].files[0].hunks.len(), 1); commit( project_repository, branch1_id, "second commit to test.txt", Some(&"test.txt:16-22".parse::().unwrap()), None, false, )?; let (branches, _) = virtual_branches::list_virtual_branches(project_repository)?; let branch = &branches.iter().find(|b| b.id == branch1_id).unwrap(); assert_eq!(branch.files.len(), 0); assert_eq!(branch.commits.len(), 2); assert_eq!(branch.commits[0].files.len(), 1); assert_eq!(branch.commits[0].files[0].hunks.len(), 1); assert_eq!(branch.commits[1].files.len(), 1); assert_eq!(branch.commits[1].files[0].hunks.len(), 1); Ok(()) } #[test] fn commit_partial_by_file() -> Result<()> { let suite = Suite::default(); let Case { project_repository, project, .. } = &suite.new_case_with_files(HashMap::from([ (PathBuf::from("test.txt"), "file1\n"), (PathBuf::from("test2.txt"), "file2\n"), ])); let commit1_oid = project_repository .git_repository .head() .unwrap() .target() .unwrap(); let commit1 = project_repository .git_repository .find_commit(commit1_oid) .unwrap(); set_test_target(project_repository)?; // remove file std::fs::remove_file(Path::new(&project.path).join("test2.txt"))?; // add new file let file_path3 = Path::new("test3.txt"); std::fs::write(Path::new(&project.path).join(file_path3), "file3\n")?; let branch1_id = create_virtual_branch(project_repository, &BranchCreateRequest::default()) .expect("failed to create virtual branch") .id; // commit commit( project_repository, branch1_id, "branch1 commit", None, None, false, )?; let (branches, _) = virtual_branches::list_virtual_branches(project_repository)?; let branch1 = &branches.iter().find(|b| b.id == branch1_id).unwrap(); // branch one test.txt has just the 1st and 3rd hunks applied let commit2 = &branch1.commits[0].id; let commit2 = project_repository .git_repository .find_commit(*commit2) .expect("failed to get commit object"); let tree = commit1.tree().expect("failed to get tree"); let file_list = tree_to_file_list(&project_repository.git_repository, &tree); assert_eq!(file_list, vec!["test.txt", "test2.txt"]); // get the tree let tree = commit2.tree().expect("failed to get tree"); let file_list = tree_to_file_list(&project_repository.git_repository, &tree); assert_eq!(file_list, vec!["test.txt", "test3.txt"]); Ok(()) } #[test] fn commit_add_and_delete_files() -> Result<()> { let suite = Suite::default(); let Case { project_repository, project, .. } = &suite.new_case_with_files(HashMap::from([ (PathBuf::from("test.txt"), "file1\n"), (PathBuf::from("test2.txt"), "file2\n"), ])); let commit1_oid = project_repository .git_repository .head() .unwrap() .target() .unwrap(); let commit1 = project_repository .git_repository .find_commit(commit1_oid) .unwrap(); set_test_target(project_repository)?; // remove file std::fs::remove_file(Path::new(&project.path).join("test2.txt"))?; // add new file let file_path3 = Path::new("test3.txt"); std::fs::write(Path::new(&project.path).join(file_path3), "file3\n")?; let branch1_id = create_virtual_branch(project_repository, &BranchCreateRequest::default()) .expect("failed to create virtual branch") .id; // commit commit( project_repository, branch1_id, "branch1 commit", None, None, false, )?; let (branches, _) = virtual_branches::list_virtual_branches(project_repository)?; let branch1 = &branches.iter().find(|b| b.id == branch1_id).unwrap(); // branch one test.txt has just the 1st and 3rd hunks applied let commit2 = &branch1.commits[0].id; let commit2 = project_repository .git_repository .find_commit(*commit2) .expect("failed to get commit object"); let tree = commit1.tree().expect("failed to get tree"); let file_list = tree_to_file_list(&project_repository.git_repository, &tree); assert_eq!(file_list, vec!["test.txt", "test2.txt"]); // get the tree let tree = commit2.tree().expect("failed to get tree"); let file_list = tree_to_file_list(&project_repository.git_repository, &tree); assert_eq!(file_list, vec!["test.txt", "test3.txt"]); Ok(()) } #[test] #[cfg(target_family = "unix")] fn commit_executable_and_symlinks() -> Result<()> { let suite = Suite::default(); let Case { project_repository, project, .. } = &suite.new_case_with_files(HashMap::from([ (PathBuf::from("test.txt"), "file1\n"), (PathBuf::from("test2.txt"), "file2\n"), ])); set_test_target(project_repository)?; // add symlinked file let file_path3 = Path::new("test3.txt"); let src = Path::new(&project.path).join("test2.txt"); let dst = Path::new(&project.path).join(file_path3); symlink(src, dst)?; // add executable let file_path4 = Path::new("test4.bin"); let exec = Path::new(&project.path).join(file_path4); std::fs::write(&exec, "exec\n")?; let permissions = std::fs::metadata(&exec)?.permissions(); let new_permissions = Permissions::from_mode(permissions.mode() | 0o111); // Add execute permission std::fs::set_permissions(&exec, new_permissions)?; let branch1_id = create_virtual_branch(project_repository, &BranchCreateRequest::default()) .expect("failed to create virtual branch") .id; // commit commit( project_repository, branch1_id, "branch1 commit", None, None, false, )?; let (branches, _) = virtual_branches::list_virtual_branches(project_repository)?; let branch1 = &branches.iter().find(|b| b.id == branch1_id).unwrap(); let commit = &branch1.commits[0].id; let commit = project_repository .git_repository .find_commit(*commit) .expect("failed to get commit object"); let tree = commit.tree().expect("failed to get tree"); let list = tree_to_entry_list(&project_repository.git_repository, &tree); assert_eq!(list[0].0, "test.txt"); assert_eq!(list[0].1, "100644"); assert_eq!(list[1].0, "test2.txt"); assert_eq!(list[1].1, "100644"); assert_eq!(list[2].0, "test3.txt"); assert_eq!(list[2].1, "120000"); assert_eq!(list[2].2, "test2.txt"); assert_eq!(list[3].0, "test4.bin"); assert_eq!(list[3].1, "100755"); Ok(()) } fn tree_to_file_list(repository: &git::Repository, tree: &git2::Tree) -> Vec { let mut file_list = Vec::new(); walk(tree, |_, entry| { let path = entry.name().unwrap(); let entry = tree.get_path(Path::new(path)).unwrap(); let object = entry.to_object(repository.into()).unwrap(); if object.kind() == Some(git2::ObjectType::Blob) { file_list.push(path.to_string()); } TreeWalkResult::Continue }) .expect("failed to walk tree"); file_list } fn tree_to_entry_list( repository: &git::Repository, tree: &git2::Tree, ) -> Vec<(String, String, String, String)> { let mut file_list = Vec::new(); walk(tree, |_root, entry| { let path = entry.name().unwrap(); let entry = tree.get_path(Path::new(path)).unwrap(); let object = entry.to_object(repository.into()).unwrap(); let blob = object.as_blob().expect("failed to get blob"); // convert content to string let octal_mode = format!("{:o}", entry.filemode()); if let Ok(content) = std::str::from_utf8(blob.content()).context("failed to convert content to string") { file_list.push(( path.to_string(), octal_mode, content.to_string(), blob.id().to_string(), )); } else { file_list.push(( path.to_string(), octal_mode, "BINARY".to_string(), blob.id().to_string(), )); } TreeWalkResult::Continue }) .expect("failed to walk tree"); file_list } #[test] fn verify_branch_commits_to_integration() -> Result<()> { let suite = Suite::default(); let Case { project_repository, project, .. } = &suite.new_case(); set_test_target(project_repository)?; verify_branch(project_repository).unwrap(); // write two commits let file_path2 = Path::new("test2.txt"); std::fs::write(Path::new(&project.path).join(file_path2), "file")?; commit_all(&project_repository.git_repository); std::fs::write(Path::new(&project.path).join(file_path2), "update")?; commit_all(&project_repository.git_repository); // verify puts commits onto the virtual branch verify_branch(project_repository).unwrap(); // one virtual branch with two commits was created let (virtual_branches, _) = virtual_branches::list_virtual_branches(project_repository)?; assert_eq!(virtual_branches.len(), 1); let branch = &virtual_branches.first().unwrap(); assert_eq!(branch.commits.len(), 2); assert_eq!(branch.commits.len(), 2); Ok(()) } #[test] fn verify_branch_not_integration() -> Result<()> { let suite = Suite::default(); let Case { project_repository, .. } = &suite.new_case(); set_test_target(project_repository)?; verify_branch(project_repository).unwrap(); project_repository .git_repository .set_head(&"refs/heads/master".parse().unwrap())?; let verify_result = verify_branch(project_repository); assert!(verify_result.is_err()); assert_eq!( verify_result.unwrap_err().to_string(), "head is refs/heads/master" ); Ok(()) } #[test] fn pre_commit_hook_rejection() -> Result<()> { let suite = Suite::default(); let Case { project, project_repository, .. } = &suite.new_case_with_files(HashMap::from([ (PathBuf::from("test.txt"), "line1\nline2\nline3\nline4\n"), (PathBuf::from("test2.txt"), "line5\nline6\nline7\nline8\n"), ])); set_test_target(project_repository)?; let branch1_id = create_virtual_branch(project_repository, &BranchCreateRequest::default()) .expect("failed to create virtual branch") .id; std::fs::write( Path::new(&project.path).join("test.txt"), "line0\nline1\nline2\nline3\nline4\n", )?; let hook = b"#!/bin/sh echo 'rejected' exit 1 "; git2_hooks::create_hook( (&project_repository.git_repository).into(), git2_hooks::HOOK_PRE_COMMIT, hook, ); let res = commit( project_repository, branch1_id, "test commit", None, None, true, ); let error = res.unwrap_err(); assert!(matches!(error, CommitError::CommitHookRejected(_))); let CommitError::CommitHookRejected(output) = error else { unreachable!() }; assert_eq!(&output, "rejected\n"); Ok(()) } #[test] fn post_commit_hook() -> Result<()> { let suite = Suite::default(); let Case { project, project_repository, .. } = &suite.new_case_with_files(HashMap::from([ (PathBuf::from("test.txt"), "line1\nline2\nline3\nline4\n"), (PathBuf::from("test2.txt"), "line5\nline6\nline7\nline8\n"), ])); set_test_target(project_repository)?; let branch1_id = create_virtual_branch(project_repository, &BranchCreateRequest::default()) .expect("failed to create virtual branch") .id; std::fs::write( Path::new(&project.path).join("test.txt"), "line0\nline1\nline2\nline3\nline4\n", )?; let hook = b"#!/bin/sh touch hook_ran "; git2_hooks::create_hook( (&project_repository.git_repository).into(), git2_hooks::HOOK_POST_COMMIT, hook, ); let hook_ran_proof = project_repository .git_repository .path() .parent() .unwrap() .join("hook_ran"); assert!(!hook_ran_proof.exists()); commit( project_repository, branch1_id, "test commit", None, None, true, )?; assert!(hook_ran_proof.exists()); Ok(()) } #[test] fn commit_msg_hook_rejection() -> Result<()> { let suite = Suite::default(); let Case { project, project_repository, .. } = &suite.new_case_with_files(HashMap::from([ (PathBuf::from("test.txt"), "line1\nline2\nline3\nline4\n"), (PathBuf::from("test2.txt"), "line5\nline6\nline7\nline8\n"), ])); set_test_target(project_repository)?; let branch1_id = create_virtual_branch(project_repository, &BranchCreateRequest::default()) .expect("failed to create virtual branch") .id; std::fs::write( Path::new(&project.path).join("test.txt"), "line0\nline1\nline2\nline3\nline4\n", )?; let hook = b"#!/bin/sh echo 'rejected' exit 1 "; git2_hooks::create_hook( (&project_repository.git_repository).into(), git2_hooks::HOOK_COMMIT_MSG, hook, ); let res = commit( project_repository, branch1_id, "test commit", None, None, true, ); let error = res.unwrap_err(); assert!(matches!(error, CommitError::CommitMsgHookRejected(_))); let CommitError::CommitMsgHookRejected(output) = error else { unreachable!() }; assert_eq!(&output, "rejected\n"); Ok(()) } fn walk(tree: &git2::Tree, mut callback: C) -> Result<()> where C: FnMut(&str, &TreeEntry) -> TreeWalkResult, { tree.walk(git2::TreeWalkMode::PreOrder, |root, entry| { match callback(root, &entry.clone()) { TreeWalkResult::Continue => git2::TreeWalkResult::Ok, // TreeWalkResult::Skip => git2::TreeWalkResult::Skip, // TreeWalkResult::Stop => git2::TreeWalkResult::Abort, } }) .map_err(Into::into) } enum TreeWalkResult { Continue, // Skip, // Stop, }