ok/jj
1
0
Fork 0
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:
Martin von Zweigbergk 2023-05-31 11:02:23 -07:00 committed by Martin von Zweigbergk
parent 1fdc25fe45
commit f4499aa65e
3 changed files with 177 additions and 47 deletions

View file

@ -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.

View file

@ -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);
} }
} }

View file

@ -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();