forked from mirrors/jj
conflicts: fix bug when modifying modify/delete conflicts
Currently, if the user modifies a modify/delete conflict, we always consider the result resolved. That happens because we materialize the missing side of the conflict as an empty string but when we parse the conflict, we expect only the number of sides in the input conflict. For example, if the input is a regular modify/delete conflict with one remove and one add, the materialized markers will have one remove and two adds (one of them empty), but when we try to parse it, we expect one remove and only one add. When we fail to parse it, we consider it resolved. This commit fixes the bug by using `conflicts::Conflict<Option<TreeValue>>` and keeping track of which sides were supposed to be empty. We could have fixed the bug without switching to `conflicts::Conflict`, but we want to switch anyway, and the fix happens naturally when switching.
This commit is contained in:
parent
1fdc25fe45
commit
f4499aa65e
3 changed files with 177 additions and 47 deletions
|
@ -99,6 +99,9 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
|
||||||
* Modify/delete conflicts now include context lines
|
* Modify/delete conflicts now include context lines
|
||||||
[#1244](https://github.com/martinvonz/jj/issues/1244).
|
[#1244](https://github.com/martinvonz/jj/issues/1244).
|
||||||
|
|
||||||
|
* It is now possible to modify either side of a modify/delete conflict (any
|
||||||
|
change used to be considered a resolution).
|
||||||
|
|
||||||
* Fixed a bug that could get partially resolved conflicts to be interpreted
|
* Fixed a bug that could get partially resolved conflicts to be interpreted
|
||||||
incorrectly.
|
incorrectly.
|
||||||
|
|
||||||
|
|
|
@ -47,6 +47,14 @@ impl<T> Conflict<T> {
|
||||||
pub fn adds(&self) -> &[T] {
|
pub fn adds(&self) -> &[T] {
|
||||||
&self.adds
|
&self.adds
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn set_remove(&mut self, i: usize, value: T) {
|
||||||
|
self.removes[i] = value;
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn set_add(&mut self, i: usize, value: T) {
|
||||||
|
self.adds[i] = value;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Conflict<Option<TreeValue>> {
|
impl Conflict<Option<TreeValue>> {
|
||||||
|
@ -68,6 +76,29 @@ impl Conflict<Option<TreeValue>> {
|
||||||
adds.resize(num_diffs + 1, None);
|
adds.resize(num_diffs + 1, None);
|
||||||
Conflict { removes, adds }
|
Conflict { removes, adds }
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Creates a `backend::Conflict` from a `Conflict` by dropping `None`
|
||||||
|
/// values. Note that the conflict is lossy: the order of `None` values is
|
||||||
|
/// not preserved when converting back to a `Conflict`.
|
||||||
|
pub fn to_backend_conflict(&self) -> backend::Conflict {
|
||||||
|
let removes = self
|
||||||
|
.removes
|
||||||
|
.iter()
|
||||||
|
.flatten()
|
||||||
|
.map(|value| backend::ConflictTerm {
|
||||||
|
value: value.clone(),
|
||||||
|
})
|
||||||
|
.collect_vec();
|
||||||
|
let adds = self
|
||||||
|
.adds
|
||||||
|
.iter()
|
||||||
|
.flatten()
|
||||||
|
.map(|value| backend::ConflictTerm {
|
||||||
|
value: value.clone(),
|
||||||
|
})
|
||||||
|
.collect_vec();
|
||||||
|
backend::Conflict { removes, adds }
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn describe_conflict_term(value: &TreeValue) -> String {
|
fn describe_conflict_term(value: &TreeValue) -> String {
|
||||||
|
@ -202,18 +233,18 @@ pub fn extract_file_conflict_as_single_hunk(
|
||||||
if file_removes.len() != conflict.removes().len() || file_adds.len() != conflict.adds().len() {
|
if file_removes.len() != conflict.removes().len() || file_adds.len() != conflict.adds().len() {
|
||||||
return None;
|
return None;
|
||||||
}
|
}
|
||||||
let removed_content = file_removes
|
let removes_content = file_removes
|
||||||
.iter()
|
.iter()
|
||||||
.map(|term| get_file_contents(store, path, *term))
|
.map(|term| get_file_contents(store, path, *term))
|
||||||
.collect_vec();
|
.collect_vec();
|
||||||
let added_content = file_adds
|
let adds_content = file_adds
|
||||||
.iter()
|
.iter()
|
||||||
.map(|term| get_file_contents(store, path, *term))
|
.map(|term| get_file_contents(store, path, *term))
|
||||||
.collect_vec();
|
.collect_vec();
|
||||||
|
|
||||||
Some(ConflictHunk {
|
Some(ConflictHunk {
|
||||||
removes: removed_content,
|
removes: removes_content,
|
||||||
adds: added_content,
|
adds: adds_content,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -414,7 +445,7 @@ pub fn update_conflict_from_content(
|
||||||
conflict_id: &ConflictId,
|
conflict_id: &ConflictId,
|
||||||
content: &[u8],
|
content: &[u8],
|
||||||
) -> BackendResult<Option<ConflictId>> {
|
) -> BackendResult<Option<ConflictId>> {
|
||||||
let mut conflict = store.read_conflict(path, conflict_id)?;
|
let conflict = store.read_conflict(path, conflict_id)?;
|
||||||
|
|
||||||
// First check if the new content is unchanged compared to the old content. If
|
// First check if the new content is unchanged compared to the old content. If
|
||||||
// it is, we don't need parse the content or write any new objects to the
|
// it is, we don't need parse the content or write any new objects to the
|
||||||
|
@ -427,9 +458,13 @@ pub fn update_conflict_from_content(
|
||||||
return Ok(Some(conflict_id.clone()));
|
return Ok(Some(conflict_id.clone()));
|
||||||
}
|
}
|
||||||
|
|
||||||
let mut removed_content = vec![vec![]; conflict.removes.len()];
|
let mut conflict = Conflict::from_backend_conflict(&conflict);
|
||||||
let mut added_content = vec![vec![]; conflict.adds.len()];
|
// TODO: Check that the conflict only involves files and convert it to a
|
||||||
if let Some(hunks) = parse_conflict(content, conflict.removes.len(), conflict.adds.len()) {
|
// `Conflict<Option<FileId>>` so we can remove the wildcard pattern in the loops
|
||||||
|
// further down.
|
||||||
|
let mut removed_content = vec![vec![]; conflict.removes().len()];
|
||||||
|
let mut added_content = vec![vec![]; conflict.adds().len()];
|
||||||
|
if let Some(hunks) = parse_conflict(content, conflict.removes().len(), conflict.adds().len()) {
|
||||||
for hunk in hunks {
|
for hunk in hunks {
|
||||||
match hunk {
|
match hunk {
|
||||||
MergeHunk::Resolved(slice) => {
|
MergeHunk::Resolved(slice) => {
|
||||||
|
@ -454,25 +489,48 @@ pub fn update_conflict_from_content(
|
||||||
// with conflict markers. Update the Conflict object with the new
|
// with conflict markers. Update the Conflict object with the new
|
||||||
// FileIds.
|
// FileIds.
|
||||||
for (i, buf) in removed_content.iter().enumerate() {
|
for (i, buf) in removed_content.iter().enumerate() {
|
||||||
let file_id = store.write_file(path, &mut buf.as_slice())?;
|
match &conflict.removes()[i] {
|
||||||
if let TreeValue::File { id, executable: _ } = &mut conflict.removes[i].value {
|
Some(TreeValue::File { id: _, executable }) => {
|
||||||
*id = file_id;
|
let file_id = store.write_file(path, &mut buf.as_slice())?;
|
||||||
} else {
|
let new_value = TreeValue::File {
|
||||||
// TODO: This can actually happen. We should check earlier
|
id: file_id,
|
||||||
// that the we only attempt to parse the conflicts if it's a
|
executable: *executable,
|
||||||
// file-only conflict.
|
};
|
||||||
panic!("Found conflict markers in merge of non-files");
|
conflict.set_remove(i, Some(new_value));
|
||||||
|
}
|
||||||
|
None if buf.is_empty() => {
|
||||||
|
// The missing side of a conflict is still represented by
|
||||||
|
// the empty string we materialized it as => nothing to do
|
||||||
|
}
|
||||||
|
_ => {
|
||||||
|
// The user edited a non-file side. This should never happen. We consider the
|
||||||
|
// conflict resolved for now.
|
||||||
|
return Ok(None);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
for (i, buf) in added_content.iter().enumerate() {
|
for (i, buf) in added_content.iter().enumerate() {
|
||||||
let file_id = store.write_file(path, &mut buf.as_slice())?;
|
match &conflict.adds()[i] {
|
||||||
if let TreeValue::File { id, executable: _ } = &mut conflict.adds[i].value {
|
Some(TreeValue::File { id: _, executable }) => {
|
||||||
*id = file_id;
|
let file_id = store.write_file(path, &mut buf.as_slice())?;
|
||||||
} else {
|
let new_value = TreeValue::File {
|
||||||
panic!("Found conflict markers in merge of non-files");
|
id: file_id,
|
||||||
|
executable: *executable,
|
||||||
|
};
|
||||||
|
conflict.set_add(i, Some(new_value));
|
||||||
|
}
|
||||||
|
None if buf.is_empty() => {
|
||||||
|
// The missing side of a conflict is still represented by
|
||||||
|
// the empty string we materialized it as => nothing to do
|
||||||
|
}
|
||||||
|
_ => {
|
||||||
|
// The user edited a non-file side. This should never happen. We consider the
|
||||||
|
// conflict resolved for now.
|
||||||
|
return Ok(None);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
let new_conflict_id = store.write_conflict(path, &conflict)?;
|
let new_conflict_id = store.write_conflict(path, &conflict.to_backend_conflict())?;
|
||||||
Ok(Some(new_conflict_id))
|
Ok(Some(new_conflict_id))
|
||||||
} else {
|
} else {
|
||||||
Ok(None)
|
Ok(None)
|
||||||
|
@ -482,74 +540,93 @@ pub fn update_conflict_from_content(
|
||||||
#[cfg(test)]
|
#[cfg(test)]
|
||||||
mod tests {
|
mod tests {
|
||||||
use super::*;
|
use super::*;
|
||||||
use crate::backend::{ConflictTerm, FileId, ObjectId};
|
use crate::backend::{FileId, ObjectId};
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn test_from_backend_conflict() {
|
fn test_backend_conflict_conversion() {
|
||||||
fn value(hex: &str) -> TreeValue {
|
fn value(hex: &str) -> TreeValue {
|
||||||
TreeValue::File {
|
TreeValue::File {
|
||||||
id: FileId::from_hex(hex),
|
id: FileId::from_hex(hex),
|
||||||
executable: false,
|
executable: false,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
fn term(hex: &str) -> ConflictTerm {
|
fn term(hex: &str) -> backend::ConflictTerm {
|
||||||
ConflictTerm { value: value(hex) }
|
backend::ConflictTerm { value: value(hex) }
|
||||||
}
|
}
|
||||||
|
|
||||||
|
let test_roundtrip = |backend_conflict: &backend::Conflict| {
|
||||||
|
let conflict = Conflict::from_backend_conflict(backend_conflict);
|
||||||
|
assert_eq!(conflict.to_backend_conflict(), *backend_conflict);
|
||||||
|
assert_eq!(
|
||||||
|
Conflict::from_backend_conflict(&conflict.to_backend_conflict()),
|
||||||
|
conflict
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
// Regular 3-way conflict
|
// Regular 3-way conflict
|
||||||
|
let backend_conflict = backend::Conflict {
|
||||||
|
removes: vec![term("11")],
|
||||||
|
adds: vec![term("22"), term("33")],
|
||||||
|
};
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
Conflict::from_backend_conflict(&backend::Conflict {
|
Conflict::from_backend_conflict(&backend_conflict),
|
||||||
removes: vec![term("11")],
|
|
||||||
adds: vec![term("22"), term("33")],
|
|
||||||
}),
|
|
||||||
Conflict {
|
Conflict {
|
||||||
removes: vec![Some(value("11"))],
|
removes: vec![Some(value("11"))],
|
||||||
adds: vec![Some(value("22")), Some(value("33"))],
|
adds: vec![Some(value("22")), Some(value("33"))],
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
test_roundtrip(&backend_conflict);
|
||||||
// Modify/delete conflict
|
// Modify/delete conflict
|
||||||
|
let backend_conflict = backend::Conflict {
|
||||||
|
removes: vec![term("11")],
|
||||||
|
adds: vec![term("22")],
|
||||||
|
};
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
Conflict::from_backend_conflict(&backend::Conflict {
|
Conflict::from_backend_conflict(&backend_conflict),
|
||||||
removes: vec![term("11")],
|
|
||||||
adds: vec![term("22")],
|
|
||||||
}),
|
|
||||||
Conflict {
|
Conflict {
|
||||||
removes: vec![Some(value("11"))],
|
removes: vec![Some(value("11"))],
|
||||||
adds: vec![Some(value("22")), None],
|
adds: vec![Some(value("22")), None],
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
test_roundtrip(&backend_conflict);
|
||||||
// Add/add conflict
|
// Add/add conflict
|
||||||
|
let backend_conflict = backend::Conflict {
|
||||||
|
removes: vec![],
|
||||||
|
adds: vec![term("11"), term("22")],
|
||||||
|
};
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
Conflict::from_backend_conflict(&backend::Conflict {
|
Conflict::from_backend_conflict(&backend_conflict),
|
||||||
removes: vec![],
|
|
||||||
adds: vec![term("11"), term("22")],
|
|
||||||
}),
|
|
||||||
Conflict {
|
Conflict {
|
||||||
removes: vec![None],
|
removes: vec![None],
|
||||||
adds: vec![Some(value("11")), Some(value("22"))],
|
adds: vec![Some(value("11")), Some(value("22"))],
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
test_roundtrip(&backend_conflict);
|
||||||
// 5-way conflict
|
// 5-way conflict
|
||||||
|
let backend_conflict = backend::Conflict {
|
||||||
|
removes: vec![term("11"), term("22")],
|
||||||
|
adds: vec![term("33"), term("44"), term("55")],
|
||||||
|
};
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
Conflict::from_backend_conflict(&backend::Conflict {
|
Conflict::from_backend_conflict(&backend_conflict),
|
||||||
removes: vec![term("11"), term("22")],
|
|
||||||
adds: vec![term("33"), term("44"), term("55")],
|
|
||||||
}),
|
|
||||||
Conflict {
|
Conflict {
|
||||||
removes: vec![Some(value("11")), Some(value("22"))],
|
removes: vec![Some(value("11")), Some(value("22"))],
|
||||||
adds: vec![Some(value("33")), Some(value("44")), Some(value("55"))],
|
adds: vec![Some(value("33")), Some(value("44")), Some(value("55"))],
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
test_roundtrip(&backend_conflict);
|
||||||
// 5-way delete/delete conflict
|
// 5-way delete/delete conflict
|
||||||
|
let backend_conflict = backend::Conflict {
|
||||||
|
removes: vec![term("11"), term("22")],
|
||||||
|
adds: vec![],
|
||||||
|
};
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
Conflict::from_backend_conflict(&backend::Conflict {
|
Conflict::from_backend_conflict(&backend_conflict),
|
||||||
removes: vec![term("11"), term("22")],
|
|
||||||
adds: vec![],
|
|
||||||
}),
|
|
||||||
Conflict {
|
Conflict {
|
||||||
removes: vec![Some(value("11")), Some(value("22"))],
|
removes: vec![Some(value("11")), Some(value("22"))],
|
||||||
adds: vec![None, None, None],
|
adds: vec![None, None, None],
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
test_roundtrip(&backend_conflict);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -650,7 +650,7 @@ fn test_update_conflict_from_content() {
|
||||||
let result = update_conflict_from_content(store, &path, &conflict_id, &materialized).unwrap();
|
let result = update_conflict_from_content(store, &path, &conflict_id, &materialized).unwrap();
|
||||||
assert_eq!(result, Some(conflict_id.clone()));
|
assert_eq!(result, Some(conflict_id.clone()));
|
||||||
|
|
||||||
// If the conflict is resolved, we None back to indicate that.
|
// If the conflict is resolved, we get None back to indicate that.
|
||||||
let result = update_conflict_from_content(
|
let result = update_conflict_from_content(
|
||||||
store,
|
store,
|
||||||
&path,
|
&path,
|
||||||
|
@ -687,6 +687,56 @@ fn test_update_conflict_from_content() {
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_update_conflict_from_content_modify_delete() {
|
||||||
|
let test_repo = TestRepo::init(false);
|
||||||
|
let store = test_repo.repo.store();
|
||||||
|
|
||||||
|
let path = RepoPath::from_internal_string("dir/file");
|
||||||
|
let before_file_id = testutils::write_file(store, &path, "line 1\nline 2 before\nline 3\n");
|
||||||
|
let after_file_id = testutils::write_file(store, &path, "line 1\nline 2 after\nline 3\n");
|
||||||
|
let conflict = Conflict {
|
||||||
|
removes: vec![file_conflict_term(&before_file_id)],
|
||||||
|
adds: vec![file_conflict_term(&after_file_id)],
|
||||||
|
};
|
||||||
|
let conflict_id = store.write_conflict(&path, &conflict).unwrap();
|
||||||
|
|
||||||
|
// If the content is unchanged compared to the materialized value, we get the
|
||||||
|
// old conflict id back.
|
||||||
|
let mut materialized = vec![];
|
||||||
|
materialize_conflict(store, &path, &conflict, &mut materialized).unwrap();
|
||||||
|
let result = update_conflict_from_content(store, &path, &conflict_id, &materialized).unwrap();
|
||||||
|
assert_eq!(result, Some(conflict_id.clone()));
|
||||||
|
|
||||||
|
// If the conflict is resolved, we get None back to indicate that.
|
||||||
|
let result = update_conflict_from_content(store, &path, &conflict_id, b"resolved\n").unwrap();
|
||||||
|
assert_eq!(result, None);
|
||||||
|
|
||||||
|
// If the conflict is modified, we get a new conflict back.
|
||||||
|
let result = update_conflict_from_content(
|
||||||
|
store,
|
||||||
|
&path,
|
||||||
|
&conflict_id,
|
||||||
|
b"<<<<<<<\n%%%%%%%\n line 1\n-line 2 before\n+line 2 modified after\n line 3\n+++++++\n>>>>>>>\n",
|
||||||
|
)
|
||||||
|
.unwrap();
|
||||||
|
assert_ne!(result, None);
|
||||||
|
assert_ne!(result, Some(conflict_id));
|
||||||
|
let new_conflict = store.read_conflict(&path, &result.unwrap()).unwrap();
|
||||||
|
// Calculate expected new FileIds
|
||||||
|
let new_base_file_id = testutils::write_file(store, &path, "line 1\nline 2 before\nline 3\n");
|
||||||
|
let new_left_file_id =
|
||||||
|
testutils::write_file(store, &path, "line 1\nline 2 modified after\nline 3\n");
|
||||||
|
|
||||||
|
assert_eq!(
|
||||||
|
new_conflict,
|
||||||
|
Conflict {
|
||||||
|
removes: vec![file_conflict_term(&new_base_file_id)],
|
||||||
|
adds: vec![file_conflict_term(&new_left_file_id)]
|
||||||
|
}
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
fn materialize_conflict_string(store: &Store, path: &RepoPath, conflict: &Conflict) -> String {
|
fn materialize_conflict_string(store: &Store, path: &RepoPath, conflict: &Conflict) -> String {
|
||||||
let mut result: Vec<u8> = vec![];
|
let mut result: Vec<u8> = vec![];
|
||||||
materialize_conflict(store, path, conflict, &mut result).unwrap();
|
materialize_conflict(store, path, conflict, &mut result).unwrap();
|
||||||
|
|
Loading…
Reference in a new issue