Merge pull request #4311 from gitbutlerapp/remove-old-locking

Remove old locking
This commit is contained in:
Caleb Owens 2024-07-09 20:39:19 +02:00 committed by GitHub
commit ddc9b33f77
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
5 changed files with 9 additions and 283 deletions

View File

@ -88,17 +88,10 @@ pub struct Project {
pub omit_certificate_check: Option<bool>, pub omit_certificate_check: Option<bool>,
// The number of changed lines that will trigger a snapshot // The number of changed lines that will trigger a snapshot
pub snapshot_lines_threshold: Option<usize>, pub snapshot_lines_threshold: Option<usize>,
#[serde(default = "default_true")]
pub use_new_locking: bool,
#[serde(default)] #[serde(default)]
pub ignore_project_semaphore: bool, pub ignore_project_semaphore: bool,
} }
fn default_true() -> bool {
true
}
impl Project { impl Project {
pub fn is_sync_enabled(&self) -> bool { pub fn is_sync_enabled(&self) -> bool {
self.api.as_ref().map(|api| api.sync).unwrap_or_default() self.api.as_ref().map(|api| api.sync).unwrap_or_default()

View File

@ -27,7 +27,6 @@ pub struct UpdateRequest {
pub omit_certificate_check: Option<bool>, pub omit_certificate_check: Option<bool>,
pub use_diff_context: Option<bool>, pub use_diff_context: Option<bool>,
pub snapshot_lines_threshold: Option<usize>, pub snapshot_lines_threshold: Option<usize>,
pub use_new_locking: Option<bool>,
pub ignore_project_semaphore: Option<bool>, pub ignore_project_semaphore: Option<bool>,
} }
@ -123,10 +122,6 @@ impl Storage {
project.snapshot_lines_threshold = Some(snapshot_lines_threshold); project.snapshot_lines_threshold = Some(snapshot_lines_threshold);
} }
if let Some(use_new_locking) = update_request.use_new_locking {
project.use_new_locking = use_new_locking;
}
if let Some(ignore_project_semaphore) = update_request.ignore_project_semaphore { if let Some(ignore_project_semaphore) = update_request.ignore_project_semaphore {
project.ignore_project_semaphore = ignore_project_semaphore; project.ignore_project_semaphore = ignore_project_semaphore;
} }

View File

@ -679,7 +679,6 @@ impl<'l> BranchManager<'l> {
let (applied_statuses, _) = get_applied_status( let (applied_statuses, _) = get_applied_status(
self.project_repository, self.project_repository,
&integration_commit.id(), &integration_commit.id(),
&target_commit.id(),
virtual_branches, virtual_branches,
) )
.context("failed to get status by branch")?; .context("failed to get status by branch")?;

View File

@ -24,7 +24,6 @@ use std::{
use anyhow::{anyhow, bail, Context, Result}; use anyhow::{anyhow, bail, Context, Result};
use bstr::{BString, ByteSlice, ByteVec}; use bstr::{BString, ByteSlice, ByteVec};
use diffy::{apply_bytes as diffy_apply, Line, Patch}; use diffy::{apply_bytes as diffy_apply, Line, Patch};
use git2::ErrorCode;
use git2_hooks::HookResult; use git2_hooks::HookResult;
use hex::ToHex; use hex::ToHex;
use serde::{Deserialize, Serialize}; use serde::{Deserialize, Serialize};
@ -206,7 +205,6 @@ pub fn unapply_ownership(
project_repository.assure_resolved()?; project_repository.assure_resolved()?;
let vb_state = project_repository.project().virtual_branches(); let vb_state = project_repository.project().virtual_branches();
let default_target = vb_state.get_default_target()?;
let virtual_branches = vb_state let virtual_branches = vb_state
.list_branches_in_workspace() .list_branches_in_workspace()
@ -214,12 +212,8 @@ pub fn unapply_ownership(
let integration_commit_id = get_workspace_head(&vb_state, project_repository)?; let integration_commit_id = get_workspace_head(&vb_state, project_repository)?;
let (applied_statuses, _) = get_applied_status( let (applied_statuses, _) =
project_repository, get_applied_status(project_repository, &integration_commit_id, virtual_branches)
&integration_commit_id,
&default_target.sha,
virtual_branches,
)
.context("failed to get status by branch")?; .context("failed to get status by branch")?;
let hunks_to_unapply = applied_statuses let hunks_to_unapply = applied_statuses
@ -1064,7 +1058,6 @@ pub fn get_status_by_branch(
project_repository, project_repository,
// TODO: Keep this optional or update lots of tests? // TODO: Keep this optional or update lots of tests?
integration_commit.unwrap_or(&default_target.sha), integration_commit.unwrap_or(&default_target.sha),
&default_target.sha,
virtual_branches, virtual_branches,
)?; )?;
@ -1144,95 +1137,11 @@ fn new_compute_locks(
Ok(locked_hunks) Ok(locked_hunks)
} }
fn compute_merge_base(
project_repository: &ProjectRepository,
target_sha: &git2::Oid,
virtual_branches: &Vec<branch::Branch>,
) -> Result<git2::Oid> {
let repo = project_repository.repo();
let mut merge_base = *target_sha;
for branch in virtual_branches {
if let Some(last) = project_repository
.l(branch.head, LogUntil::Commit(*target_sha))?
.last()
.map(|id| repo.find_commit(id.to_owned()))
{
if let Ok(parent) = last?.parent(0) {
merge_base = repo.merge_base(parent.id(), merge_base)?;
}
}
}
Ok(merge_base)
}
fn compute_locks(
project_repository: &ProjectRepository,
integration_commit: &git2::Oid,
target_sha: &git2::Oid,
base_diffs: &BranchStatus,
virtual_branches: &Vec<branch::Branch>,
) -> Result<HashMap<HunkHash, Vec<diff::HunkLock>>> {
let merge_base = compute_merge_base(project_repository, target_sha, virtual_branches)?;
let mut locked_hunk_map = HashMap::<HunkHash, Vec<diff::HunkLock>>::new();
let mut commit_to_branch = HashMap::new();
for branch in virtual_branches {
for commit in project_repository.log(branch.head, LogUntil::Commit(*target_sha))? {
commit_to_branch.insert(commit.id(), branch.id);
}
}
for (path, hunks) in base_diffs.clone().into_iter() {
for hunk in hunks {
let blame = match project_repository.repo().blame(
&path,
hunk.old_start,
(hunk.old_start + hunk.old_lines).saturating_sub(1),
merge_base,
*integration_commit,
) {
Ok(blame) => blame,
Err(error) => {
if error.code() == ErrorCode::NotFound {
continue;
} else {
return Err(error.into());
}
}
};
for blame_hunk in blame.iter() {
let commit_id = blame_hunk.orig_commit_id();
if commit_id == *integration_commit || commit_id == *target_sha {
continue;
}
let hash = Hunk::hash_diff(&hunk.diff_lines);
let Some(branch_id) = commit_to_branch.get(&commit_id) else {
continue;
};
let hunk_lock = diff::HunkLock {
branch_id: *branch_id,
commit_id,
};
locked_hunk_map
.entry(hash)
.and_modify(|locks| {
locks.push(hunk_lock);
})
.or_insert(vec![hunk_lock]);
}
}
}
Ok(locked_hunk_map)
}
// Returns branches and their associated file changes, in addition to a list // Returns branches and their associated file changes, in addition to a list
// of skipped files. // of skipped files.
pub(crate) fn get_applied_status( pub(crate) fn get_applied_status(
project_repository: &ProjectRepository, project_repository: &ProjectRepository,
integration_commit: &git2::Oid, integration_commit: &git2::Oid,
target_sha: &git2::Oid,
mut virtual_branches: Vec<branch::Branch>, mut virtual_branches: Vec<branch::Branch>,
) -> Result<(AppliedStatuses, Vec<diff::FileDiff>)> { ) -> Result<(AppliedStatuses, Vec<diff::FileDiff>)> {
let base_file_diffs = diff::workdir(project_repository.repo(), &integration_commit.to_owned()) let base_file_diffs = diff::workdir(project_repository.repo(), &integration_commit.to_owned())
@ -1262,17 +1171,7 @@ pub(crate) fn get_applied_status(
.map(|branch| (branch.id, HashMap::new())) .map(|branch| (branch.id, HashMap::new()))
.collect(); .collect();
let locks = if project_repository.project().use_new_locking { let locks = new_compute_locks(project_repository.repo(), &base_diffs, &virtual_branches)?;
new_compute_locks(project_repository.repo(), &base_diffs, &virtual_branches)?
} else {
compute_locks(
project_repository,
integration_commit,
target_sha,
&base_diffs,
&virtual_branches,
)?
};
for branch in &mut virtual_branches { for branch in &mut virtual_branches {
let old_claims = branch.ownership.claims.clone(); let old_claims = branch.ownership.claims.clone();
@ -1538,7 +1437,7 @@ pub fn write_tree_onto_commit(
files: impl IntoIterator<Item = (impl Borrow<PathBuf>, impl Borrow<Vec<diff::GitHunk>>)>, files: impl IntoIterator<Item = (impl Borrow<PathBuf>, impl Borrow<Vec<diff::GitHunk>>)>,
) -> Result<git2::Oid> { ) -> Result<git2::Oid> {
// read the base sha into an index // read the base sha into an index
let git_repository = project_repository.repo(); let git_repository: &git2::Repository = project_repository.repo();
let head_commit = git_repository.find_commit(commit_oid)?; let head_commit = git_repository.find_commit(commit_oid)?;
let base_tree = head_commit.tree()?; let base_tree = head_commit.tree()?;
@ -2248,12 +2147,8 @@ pub fn amend(
let integration_commit_id = let integration_commit_id =
crate::integration::get_workspace_head(&vb_state, project_repository)?; crate::integration::get_workspace_head(&vb_state, project_repository)?;
let (mut applied_statuses, _) = get_applied_status( let (mut applied_statuses, _) =
project_repository, get_applied_status(project_repository, &integration_commit_id, virtual_branches)?;
&integration_commit_id,
&default_target.sha,
virtual_branches,
)?;
let (ref mut target_branch, target_status) = applied_statuses let (ref mut target_branch, target_status) = applied_statuses
.iter_mut() .iter_mut()
@ -2733,17 +2628,11 @@ pub fn move_commit(
bail!("branch {target_branch_id} is not among applied branches") bail!("branch {target_branch_id} is not among applied branches")
} }
let default_target = vb_state.get_default_target()?;
let integration_commit_id = let integration_commit_id =
crate::integration::get_workspace_head(&vb_state, project_repository)?; crate::integration::get_workspace_head(&vb_state, project_repository)?;
let (mut applied_statuses, _) = get_applied_status( let (mut applied_statuses, _) =
project_repository, get_applied_status(project_repository, &integration_commit_id, applied_branches)?;
&integration_commit_id,
&default_target.sha,
applied_branches,
)?;
let (ref mut source_branch, source_status) = applied_statuses let (ref mut source_branch, source_status) = applied_statuses
.iter_mut() .iter_mut()

View File

@ -58,106 +58,6 @@ async fn should_lock_updated_hunks() {
} }
} }
#[tokio::test]
async fn should_not_lock_disjointed_hunks() {
let Test {
project,
controller,
repository,
..
} = &Test::default();
let mut lines: Vec<_> = (0_i32..24_i32).map(|i| format!("line {}", i)).collect();
repository.write_file("file.txt", &lines);
repository.commit_all("my commit");
repository.push();
controller
.set_base_branch(project, &"refs/remotes/origin/master".parse().unwrap())
.await
.unwrap();
let branch_id = controller
.create_virtual_branch(project, &branch::BranchCreateRequest::default())
.await
.unwrap();
{
// new hunk in the middle of the file
lines[12] = "commited stuff".to_string();
repository.write_file("file.txt", &lines);
let branch = get_virtual_branch(controller, project, branch_id).await;
assert_eq!(branch.files.len(), 1);
assert_eq!(branch.files[0].path.display().to_string(), "file.txt");
assert_eq!(branch.files[0].hunks.len(), 1);
assert!(!branch.files[0].hunks[0].locked);
}
controller
.create_commit(project, branch_id, "test commit", None, false)
.await
.unwrap();
controller
.push_virtual_branch(project, branch_id, false, None)
.await
.unwrap();
{
// hunk before the commited part is not locked
let mut changed_lines = lines.clone();
changed_lines[8] = "updated line".to_string();
repository.write_file("file.txt", &changed_lines);
let branch = get_virtual_branch(controller, project, branch_id).await;
assert_eq!(branch.files.len(), 1);
assert_eq!(branch.files[0].path.display().to_string(), "file.txt");
assert_eq!(branch.files[0].hunks.len(), 1);
assert!(!branch.files[0].hunks[0].locked);
// cleanup
repository.write_file("file.txt", &lines);
}
{
// hunk after the commited part is not locked
let mut changed_lines = lines.clone();
changed_lines[16] = "updated line".to_string();
repository.write_file("file.txt", &changed_lines);
let branch = get_virtual_branch(controller, project, branch_id).await;
assert_eq!(branch.files.len(), 1);
assert_eq!(branch.files[0].path.display().to_string(), "file.txt");
assert_eq!(branch.files[0].hunks.len(), 1);
assert!(!branch.files[0].hunks[0].locked);
// cleanup
repository.write_file("file.txt", &lines);
}
{
// hunk before the commited part but with overlapping context
let mut changed_lines = lines.clone();
changed_lines[10] = "updated line".to_string();
repository.write_file("file.txt", &changed_lines);
let branch = get_virtual_branch(controller, project, branch_id).await;
assert_eq!(branch.files.len(), 1);
assert_eq!(branch.files[0].path.display().to_string(), "file.txt");
assert_eq!(branch.files[0].hunks.len(), 1);
// TODO: We lock this hunk, but can we afford not lock it?
assert!(branch.files[0].hunks[0].locked);
// cleanup
repository.write_file("file.txt", &lines);
}
{
// hunk after the commited part but with overlapping context
let mut changed_lines = lines.clone();
changed_lines[14] = "updated line".to_string();
repository.write_file("file.txt", &changed_lines);
let branch = get_virtual_branch(controller, project, branch_id).await;
assert_eq!(branch.files.len(), 1);
assert_eq!(branch.files[0].path.display().to_string(), "file.txt");
assert_eq!(branch.files[0].hunks.len(), 1);
// TODO: We lock this hunk, but can we afford not lock it?
assert!(branch.files[0].hunks[0].locked);
// cleanup
repository.write_file("file.txt", &lines);
}
}
#[tokio::test] #[tokio::test]
async fn should_reset_into_same_branch() { async fn should_reset_into_same_branch() {
let Test { let Test {
@ -228,56 +128,6 @@ async fn should_reset_into_same_branch() {
assert_eq!(files.len(), 1); assert_eq!(files.len(), 1);
} }
#[tokio::test]
async fn should_double_lock() {
let Test {
project,
controller,
repository,
..
} = &Test::default();
let mut lines = repository.gen_file("file.txt", 7);
repository.write_file("file.txt", &lines);
commit_and_push_initial(repository);
controller
.set_base_branch(project, &"refs/remotes/origin/master".parse().unwrap())
.await
.unwrap();
let branch_id = controller
.create_virtual_branch(project, &branch::BranchCreateRequest::default())
.await
.unwrap();
lines[0] = "change 1".to_string();
repository.write_file("file.txt", &lines);
let commit_1 = controller
.create_commit(project, branch_id, "commit 1", None, false)
.await
.unwrap();
lines[6] = "change 2".to_string();
repository.write_file("file.txt", &lines);
let commit_2 = controller
.create_commit(project, branch_id, "commit 2", None, false)
.await
.unwrap();
lines[3] = "change3".to_string();
repository.write_file("file.txt", &lines);
let branch = get_virtual_branch(controller, project, branch_id).await;
let locks = &branch.files[0].hunks[0].locked_to.clone().unwrap();
assert_eq!(locks.len(), 2);
assert_eq!(locks[0].commit_id, commit_1);
assert_eq!(locks[1].commit_id, commit_2);
}
fn commit_and_push_initial(repository: &TestProject) { fn commit_and_push_initial(repository: &TestProject) {
repository.commit_all("initial commit"); repository.commit_all("initial commit");
repository.push(); repository.push();