mirror of
https://github.com/martinvonz/jj.git
synced 2025-01-26 06:01:48 +00:00
739bf8decf
I'm going to add "checked" version of to_fs_path(), but all callers can't be migrated to it. For example, an error message should be produced even if the path is malformed. This patch also adds error variants to propagate InvalidRepoPathError. They don't use ::Other { .. } so the errors can be distinguished in tests.
717 lines
25 KiB
Rust
717 lines
25 KiB
Rust
// Copyright 2020 The Jujutsu Authors
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// https://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
use itertools::Itertools;
|
|
use jj_lib::backend::TreeValue;
|
|
use jj_lib::repo::Repo;
|
|
use jj_lib::repo_path::RepoPath;
|
|
use jj_lib::repo_path::RepoPathComponent;
|
|
use jj_lib::rewrite::rebase_commit;
|
|
use jj_lib::tree::merge_trees;
|
|
use jj_lib::tree::Tree;
|
|
use testutils::create_single_tree;
|
|
use testutils::create_tree;
|
|
use testutils::TestRepo;
|
|
|
|
#[test]
|
|
fn test_same_type() {
|
|
// Tests all possible cases where the entry type is unchanged, specifically
|
|
// using only normal files in all trees (no symlinks, no trees, etc.).
|
|
|
|
let test_repo = TestRepo::init();
|
|
let repo = &test_repo.repo;
|
|
let store = repo.store();
|
|
|
|
// The file name encodes the state in the base and in each side ("_" means
|
|
// missing)
|
|
let files = vec![
|
|
"__a", // side 2 added
|
|
"_a_", // side 1 added
|
|
"_aa", // both sides added, same content
|
|
"_ab", // both sides added, different content
|
|
"a__", // both sides removed
|
|
"a_a", // side 1 removed
|
|
"a_b", // side 1 removed, side 2 modified
|
|
"aa_", // side 2 removed
|
|
"aaa", // no changes
|
|
"aab", // side 2 modified
|
|
"ab_", // side 1 modified, side 2 removed
|
|
"aba", // side 1 modified
|
|
"abb", // both sides modified, same content
|
|
"abc", // both sides modified, different content
|
|
];
|
|
|
|
let write_tree = |index: usize| -> Tree {
|
|
let mut tree_builder = store.tree_builder(store.empty_tree_id().clone());
|
|
for &path in &files {
|
|
let contents = &path[index..][..1];
|
|
if contents != "_" {
|
|
testutils::write_normal_file(
|
|
&mut tree_builder,
|
|
RepoPath::from_internal_string(path),
|
|
contents,
|
|
);
|
|
}
|
|
}
|
|
let tree_id = tree_builder.write_tree().unwrap();
|
|
store.get_tree(RepoPath::root(), &tree_id).unwrap()
|
|
};
|
|
|
|
let base_tree = write_tree(0);
|
|
let side1_tree = write_tree(1);
|
|
let side2_tree = write_tree(2);
|
|
|
|
// Create the merged tree
|
|
let merged_tree = merge_trees(&side1_tree, &base_tree, &side2_tree).unwrap();
|
|
|
|
// Check that we have exactly the paths we expect in the merged tree
|
|
let names = merged_tree
|
|
.entries_non_recursive()
|
|
.map(|entry| entry.name().as_internal_str())
|
|
.collect_vec();
|
|
assert_eq!(
|
|
names,
|
|
vec!["__a", "_a_", "_aa", "_ab", "a_b", "aaa", "aab", "ab_", "aba", "abb", "abc",]
|
|
);
|
|
|
|
// Check that the simple, non-conflicting cases were resolved correctly
|
|
assert_eq!(
|
|
merged_tree.value(RepoPathComponent::new("__a")),
|
|
side2_tree.value(RepoPathComponent::new("__a"))
|
|
);
|
|
assert_eq!(
|
|
merged_tree.value(RepoPathComponent::new("_a_")),
|
|
side1_tree.value(RepoPathComponent::new("_a_"))
|
|
);
|
|
assert_eq!(
|
|
merged_tree.value(RepoPathComponent::new("_aa")),
|
|
side1_tree.value(RepoPathComponent::new("_aa"))
|
|
);
|
|
assert_eq!(
|
|
merged_tree.value(RepoPathComponent::new("aaa")),
|
|
side1_tree.value(RepoPathComponent::new("aaa"))
|
|
);
|
|
assert_eq!(
|
|
merged_tree.value(RepoPathComponent::new("aab")),
|
|
side2_tree.value(RepoPathComponent::new("aab"))
|
|
);
|
|
assert_eq!(
|
|
merged_tree.value(RepoPathComponent::new("aba")),
|
|
side1_tree.value(RepoPathComponent::new("aba"))
|
|
);
|
|
assert_eq!(
|
|
merged_tree.value(RepoPathComponent::new("abb")),
|
|
side1_tree.value(RepoPathComponent::new("abb"))
|
|
);
|
|
|
|
// Check the conflicting cases
|
|
let component = RepoPathComponent::new("_ab");
|
|
match merged_tree.value(component).unwrap() {
|
|
TreeValue::Conflict(id) => {
|
|
let conflict = store
|
|
.read_conflict(RepoPath::from_internal_string("_ab"), id)
|
|
.unwrap();
|
|
assert_eq!(
|
|
conflict.adds().map(|v| v.as_ref()).collect_vec(),
|
|
vec![side1_tree.value(component), side2_tree.value(component)]
|
|
);
|
|
assert_eq!(
|
|
conflict.removes().map(|v| v.as_ref()).collect_vec(),
|
|
vec![None]
|
|
);
|
|
}
|
|
_ => panic!("unexpected value"),
|
|
};
|
|
let component = RepoPathComponent::new("a_b");
|
|
match merged_tree.value(component).unwrap() {
|
|
TreeValue::Conflict(id) => {
|
|
let conflict = store
|
|
.read_conflict(RepoPath::from_internal_string("a_b"), id)
|
|
.unwrap();
|
|
assert_eq!(
|
|
conflict.removes().map(|v| v.as_ref()).collect_vec(),
|
|
vec![base_tree.value(component)]
|
|
);
|
|
assert_eq!(
|
|
conflict.adds().map(|v| v.as_ref()).collect_vec(),
|
|
vec![side2_tree.value(component), None]
|
|
);
|
|
}
|
|
_ => panic!("unexpected value"),
|
|
};
|
|
let component = RepoPathComponent::new("ab_");
|
|
match merged_tree.value(component).unwrap() {
|
|
TreeValue::Conflict(id) => {
|
|
let conflict = store
|
|
.read_conflict(RepoPath::from_internal_string("ab_"), id)
|
|
.unwrap();
|
|
assert_eq!(
|
|
conflict.removes().map(|v| v.as_ref()).collect_vec(),
|
|
vec![base_tree.value(component)]
|
|
);
|
|
assert_eq!(
|
|
conflict.adds().map(|v| v.as_ref()).collect_vec(),
|
|
vec![side1_tree.value(component), None]
|
|
);
|
|
}
|
|
_ => panic!("unexpected value"),
|
|
};
|
|
let component = RepoPathComponent::new("abc");
|
|
match merged_tree.value(component).unwrap() {
|
|
TreeValue::Conflict(id) => {
|
|
let conflict = store
|
|
.read_conflict(RepoPath::from_internal_string("abc"), id)
|
|
.unwrap();
|
|
assert_eq!(
|
|
conflict.removes().map(|v| v.as_ref()).collect_vec(),
|
|
vec![base_tree.value(component)]
|
|
);
|
|
assert_eq!(
|
|
conflict.adds().map(|v| v.as_ref()).collect_vec(),
|
|
vec![side1_tree.value(component), side2_tree.value(component)]
|
|
);
|
|
}
|
|
_ => panic!("unexpected value"),
|
|
};
|
|
}
|
|
|
|
#[test]
|
|
fn test_executable() {
|
|
let test_repo = TestRepo::init();
|
|
let repo = &test_repo.repo;
|
|
let store = repo.store();
|
|
|
|
// The file name encodes whether the file was executable or normal in the base
|
|
// and in each side
|
|
let files = vec!["nnn", "nnx", "nxn", "nxx", "xnn", "xnx", "xxn", "xxx"];
|
|
|
|
let write_tree = |files: &[(&str, bool)]| -> Tree {
|
|
let mut tree_builder = store.tree_builder(store.empty_tree_id().clone());
|
|
for &(path, executable) in files {
|
|
let repo_path = RepoPath::from_internal_string(path);
|
|
if executable {
|
|
testutils::write_executable_file(&mut tree_builder, repo_path, "contents");
|
|
} else {
|
|
testutils::write_normal_file(&mut tree_builder, repo_path, "contents");
|
|
}
|
|
}
|
|
let tree_id = tree_builder.write_tree().unwrap();
|
|
store.get_tree(RepoPath::root(), &tree_id).unwrap()
|
|
};
|
|
|
|
fn contents_in_tree<'a>(files: &[&'a str], index: usize) -> Vec<(&'a str, bool)> {
|
|
files
|
|
.iter()
|
|
.map(|f| (*f, &f[index..][..1] == "x"))
|
|
.collect()
|
|
}
|
|
|
|
let base_tree = write_tree(&contents_in_tree(&files, 0));
|
|
let side1_tree = write_tree(&contents_in_tree(&files, 1));
|
|
let side2_tree = write_tree(&contents_in_tree(&files, 2));
|
|
|
|
// Create the merged tree
|
|
let merged_tree = merge_trees(&side1_tree, &base_tree, &side2_tree).unwrap();
|
|
|
|
// Check that the merged tree has the correct executable bits
|
|
let norm = base_tree.value(RepoPathComponent::new("nnn"));
|
|
let exec = base_tree.value(RepoPathComponent::new("xxx"));
|
|
assert_eq!(merged_tree.value(RepoPathComponent::new("nnn")), norm);
|
|
assert_eq!(merged_tree.value(RepoPathComponent::new("nnx")), exec);
|
|
assert_eq!(merged_tree.value(RepoPathComponent::new("nxn")), exec);
|
|
assert_eq!(merged_tree.value(RepoPathComponent::new("nxx")), exec);
|
|
assert_eq!(merged_tree.value(RepoPathComponent::new("xnn")), norm);
|
|
assert_eq!(merged_tree.value(RepoPathComponent::new("xnx")), norm);
|
|
assert_eq!(merged_tree.value(RepoPathComponent::new("xxn")), norm);
|
|
assert_eq!(merged_tree.value(RepoPathComponent::new("xxx")), exec);
|
|
}
|
|
|
|
#[test]
|
|
fn test_subtrees() {
|
|
// Tests that subtrees are merged.
|
|
|
|
let test_repo = TestRepo::init();
|
|
let repo = &test_repo.repo;
|
|
let store = repo.store();
|
|
|
|
let write_tree = |paths: Vec<&str>| -> Tree {
|
|
let mut tree_builder = store.tree_builder(store.empty_tree_id().clone());
|
|
for path in paths {
|
|
testutils::write_normal_file(
|
|
&mut tree_builder,
|
|
RepoPath::from_internal_string(path),
|
|
&format!("contents of {path:?}"),
|
|
);
|
|
}
|
|
let tree_id = tree_builder.write_tree().unwrap();
|
|
store.get_tree(RepoPath::root(), &tree_id).unwrap()
|
|
};
|
|
|
|
let base_tree = write_tree(vec!["f1", "d1/f1", "d1/d1/f1", "d1/d1/d1/f1"]);
|
|
let side1_tree = write_tree(vec![
|
|
"f1",
|
|
"f2",
|
|
"d1/f1",
|
|
"d1/f2",
|
|
"d1/d1/f1",
|
|
"d1/d1/d1/f1",
|
|
]);
|
|
let side2_tree = write_tree(vec![
|
|
"f1",
|
|
"d1/f1",
|
|
"d1/d1/f1",
|
|
"d1/d1/d1/f1",
|
|
"d1/d1/d1/f2",
|
|
]);
|
|
|
|
let merged_tree = merge_trees(&side1_tree, &base_tree, &side2_tree).unwrap();
|
|
let entries = merged_tree.entries().collect_vec();
|
|
|
|
let expected_tree = write_tree(vec![
|
|
"f1",
|
|
"f2",
|
|
"d1/f1",
|
|
"d1/f2",
|
|
"d1/d1/f1",
|
|
"d1/d1/d1/f1",
|
|
"d1/d1/d1/f2",
|
|
]);
|
|
let expected_entries = expected_tree.entries().collect_vec();
|
|
assert_eq!(entries, expected_entries);
|
|
}
|
|
|
|
#[test]
|
|
fn test_subtree_becomes_empty() {
|
|
// Tests that subtrees that become empty are removed from the parent tree.
|
|
|
|
let test_repo = TestRepo::init();
|
|
let repo = &test_repo.repo;
|
|
let store = repo.store();
|
|
|
|
let write_tree = |paths: Vec<&str>| -> Tree {
|
|
let mut tree_builder = store.tree_builder(store.empty_tree_id().clone());
|
|
for path in paths {
|
|
testutils::write_normal_file(
|
|
&mut tree_builder,
|
|
RepoPath::from_internal_string(path),
|
|
&format!("contents of {path:?}"),
|
|
);
|
|
}
|
|
let tree_id = tree_builder.write_tree().unwrap();
|
|
store.get_tree(RepoPath::root(), &tree_id).unwrap()
|
|
};
|
|
|
|
let base_tree = write_tree(vec!["f1", "d1/f1", "d1/d1/d1/f1", "d1/d1/d1/f2"]);
|
|
let side1_tree = write_tree(vec!["f1", "d1/f1", "d1/d1/d1/f1"]);
|
|
let side2_tree = write_tree(vec!["d1/d1/d1/f2"]);
|
|
|
|
let merged_tree = merge_trees(&side1_tree, &base_tree, &side2_tree).unwrap();
|
|
assert_eq!(merged_tree.id(), store.empty_tree_id());
|
|
}
|
|
|
|
#[test]
|
|
fn test_subtree_one_missing() {
|
|
// Tests that merging trees where one side is missing is resolved as if the
|
|
// missing side was empty.
|
|
let test_repo = TestRepo::init();
|
|
let repo = &test_repo.repo;
|
|
let store = repo.store();
|
|
|
|
let write_tree = |paths: Vec<&str>| -> Tree {
|
|
let mut tree_builder = store.tree_builder(store.empty_tree_id().clone());
|
|
for path in paths {
|
|
testutils::write_normal_file(
|
|
&mut tree_builder,
|
|
RepoPath::from_internal_string(path),
|
|
&format!("contents of {path:?}"),
|
|
);
|
|
}
|
|
let tree_id = tree_builder.write_tree().unwrap();
|
|
store.get_tree(RepoPath::root(), &tree_id).unwrap()
|
|
};
|
|
|
|
let tree1 = write_tree(vec![]);
|
|
let tree2 = write_tree(vec!["d1/f1"]);
|
|
let tree3 = write_tree(vec!["d1/f1", "d1/f2"]);
|
|
|
|
// The two sides add different trees
|
|
let merged_tree = merge_trees(&tree2, &tree1, &tree3).unwrap();
|
|
let expected_entries = write_tree(vec!["d1/f1", "d1/f2"]).entries().collect_vec();
|
|
assert_eq!(merged_tree.entries().collect_vec(), expected_entries);
|
|
// Same tree other way
|
|
let reverse_merged_tree = merge_trees(&tree3, &tree1, &tree2).unwrap();
|
|
assert_eq!(reverse_merged_tree.id(), merged_tree.id());
|
|
|
|
// One side removes, the other side modifies
|
|
let merged_tree = merge_trees(&tree1, &tree2, &tree3).unwrap();
|
|
let expected_entries = write_tree(vec!["d1/f2"]).entries().collect_vec();
|
|
assert_eq!(merged_tree.entries().collect_vec(), expected_entries);
|
|
// Same tree other way
|
|
let reverse_merged_tree = merge_trees(&tree3, &tree2, &tree1).unwrap();
|
|
assert_eq!(reverse_merged_tree.id(), merged_tree.id());
|
|
}
|
|
|
|
#[test]
|
|
fn test_types() {
|
|
// Tests conflicts between different types. This is mostly to test that the
|
|
// conflicts survive the roundtrip to the store.
|
|
|
|
let test_repo = TestRepo::init();
|
|
let repo = &test_repo.repo;
|
|
let store = repo.store();
|
|
|
|
let mut base_tree_builder = store.tree_builder(store.empty_tree_id().clone());
|
|
let mut side1_tree_builder = store.tree_builder(store.empty_tree_id().clone());
|
|
let mut side2_tree_builder = store.tree_builder(store.empty_tree_id().clone());
|
|
testutils::write_normal_file(
|
|
&mut base_tree_builder,
|
|
RepoPath::from_internal_string("normal_executable_symlink"),
|
|
"contents",
|
|
);
|
|
testutils::write_executable_file(
|
|
&mut side1_tree_builder,
|
|
RepoPath::from_internal_string("normal_executable_symlink"),
|
|
"contents",
|
|
);
|
|
testutils::write_symlink(
|
|
&mut side2_tree_builder,
|
|
RepoPath::from_internal_string("normal_executable_symlink"),
|
|
"contents",
|
|
);
|
|
let tree_id = store.empty_tree_id().clone();
|
|
base_tree_builder.set(
|
|
RepoPath::from_internal_string("tree_normal_symlink").to_owned(),
|
|
TreeValue::Tree(tree_id),
|
|
);
|
|
testutils::write_normal_file(
|
|
&mut side1_tree_builder,
|
|
RepoPath::from_internal_string("tree_normal_symlink"),
|
|
"contents",
|
|
);
|
|
testutils::write_symlink(
|
|
&mut side2_tree_builder,
|
|
RepoPath::from_internal_string("tree_normal_symlink"),
|
|
"contents",
|
|
);
|
|
let base_tree_id = base_tree_builder.write_tree().unwrap();
|
|
let base_tree = store.get_tree(RepoPath::root(), &base_tree_id).unwrap();
|
|
let side1_tree_id = side1_tree_builder.write_tree().unwrap();
|
|
let side1_tree = store.get_tree(RepoPath::root(), &side1_tree_id).unwrap();
|
|
let side2_tree_id = side2_tree_builder.write_tree().unwrap();
|
|
let side2_tree = store.get_tree(RepoPath::root(), &side2_tree_id).unwrap();
|
|
|
|
// Created the merged tree
|
|
let merged_tree = merge_trees(&side1_tree, &base_tree, &side2_tree).unwrap();
|
|
|
|
// Check the conflicting cases
|
|
let component = RepoPathComponent::new("normal_executable_symlink");
|
|
match merged_tree.value(component).unwrap() {
|
|
TreeValue::Conflict(id) => {
|
|
let conflict = store
|
|
.read_conflict(
|
|
RepoPath::from_internal_string("normal_executable_symlink"),
|
|
id,
|
|
)
|
|
.unwrap();
|
|
assert_eq!(
|
|
conflict.removes().map(|v| v.as_ref()).collect_vec(),
|
|
vec![base_tree.value(component)]
|
|
);
|
|
assert_eq!(
|
|
conflict.adds().map(|v| v.as_ref()).collect_vec(),
|
|
vec![side1_tree.value(component), side2_tree.value(component)]
|
|
);
|
|
}
|
|
_ => panic!("unexpected value"),
|
|
};
|
|
let component = RepoPathComponent::new("tree_normal_symlink");
|
|
match merged_tree.value(component).unwrap() {
|
|
TreeValue::Conflict(id) => {
|
|
let conflict = store
|
|
.read_conflict(RepoPath::from_internal_string("tree_normal_symlink"), id)
|
|
.unwrap();
|
|
assert_eq!(
|
|
conflict.removes().map(|v| v.as_ref()).collect_vec(),
|
|
vec![base_tree.value(component)]
|
|
);
|
|
assert_eq!(
|
|
conflict.adds().map(|v| v.as_ref()).collect_vec(),
|
|
vec![side1_tree.value(component), side2_tree.value(component)]
|
|
);
|
|
}
|
|
_ => panic!("unexpected value"),
|
|
};
|
|
}
|
|
|
|
#[test]
|
|
fn test_simplify_conflict() {
|
|
let test_repo = TestRepo::init();
|
|
let repo = &test_repo.repo;
|
|
let store = repo.store();
|
|
|
|
let component = RepoPathComponent::new("file");
|
|
let path = RepoPath::from_internal_string("file");
|
|
let write_tree = |contents: &str| -> Tree { create_single_tree(repo, &[(path, contents)]) };
|
|
|
|
let base_tree = write_tree("base contents");
|
|
let branch_tree = write_tree("branch contents");
|
|
let upstream1_tree = write_tree("upstream1 contents");
|
|
let upstream2_tree = write_tree("upstream2 contents");
|
|
|
|
// Rebase the branch tree to the first upstream tree
|
|
let rebased1_tree = merge_trees(&branch_tree, &base_tree, &upstream1_tree).unwrap();
|
|
// Make sure we have a conflict (testing the test setup)
|
|
match rebased1_tree.value(component).unwrap() {
|
|
TreeValue::Conflict(_) => {
|
|
// expected
|
|
}
|
|
_ => panic!("unexpected value"),
|
|
};
|
|
|
|
// Rebase the rebased tree back to the base. The conflict should be gone. Try
|
|
// both directions.
|
|
let rebased_back_tree = merge_trees(&rebased1_tree, &upstream1_tree, &base_tree).unwrap();
|
|
assert_eq!(
|
|
rebased_back_tree.value(component),
|
|
branch_tree.value(component)
|
|
);
|
|
let rebased_back_tree = merge_trees(&base_tree, &upstream1_tree, &rebased1_tree).unwrap();
|
|
assert_eq!(
|
|
rebased_back_tree.value(component),
|
|
branch_tree.value(component)
|
|
);
|
|
|
|
// Rebase the rebased tree further upstream. The conflict should be simplified
|
|
// to not mention the contents from the first rebase.
|
|
let further_rebased_tree =
|
|
merge_trees(&rebased1_tree, &upstream1_tree, &upstream2_tree).unwrap();
|
|
match further_rebased_tree.value(component).unwrap() {
|
|
TreeValue::Conflict(id) => {
|
|
let conflict = store
|
|
.read_conflict(
|
|
RepoPath::from_internal_string(component.as_internal_str()),
|
|
id,
|
|
)
|
|
.unwrap();
|
|
assert_eq!(
|
|
conflict.removes().map(|v| v.as_ref()).collect_vec(),
|
|
vec![base_tree.value(component)]
|
|
);
|
|
assert_eq!(
|
|
conflict.adds().map(|v| v.as_ref()).collect_vec(),
|
|
vec![
|
|
branch_tree.value(component),
|
|
upstream2_tree.value(component),
|
|
]
|
|
);
|
|
}
|
|
_ => panic!("unexpected value"),
|
|
};
|
|
let further_rebased_tree =
|
|
merge_trees(&upstream2_tree, &upstream1_tree, &rebased1_tree).unwrap();
|
|
match further_rebased_tree.value(component).unwrap() {
|
|
TreeValue::Conflict(id) => {
|
|
let conflict = store.read_conflict(path, id).unwrap();
|
|
assert_eq!(
|
|
conflict.removes().map(|v| v.as_ref()).collect_vec(),
|
|
vec![base_tree.value(component)]
|
|
);
|
|
assert_eq!(
|
|
conflict.adds().map(|v| v.as_ref()).collect_vec(),
|
|
vec![
|
|
upstream2_tree.value(component),
|
|
branch_tree.value(component)
|
|
]
|
|
);
|
|
}
|
|
_ => panic!("unexpected value"),
|
|
};
|
|
}
|
|
|
|
#[test]
|
|
fn test_simplify_conflict_after_resolving_parent() {
|
|
let settings = testutils::user_settings();
|
|
let test_repo = TestRepo::init();
|
|
let repo = &test_repo.repo;
|
|
|
|
// Set up a repo like this:
|
|
// D
|
|
// | C
|
|
// | B
|
|
// |/
|
|
// A
|
|
//
|
|
// Commit A has a file with 3 lines. B and D make conflicting changes to the
|
|
// first line. C changes the third line. We then rebase B and C onto D,
|
|
// which creates a conflict. We resolve the conflict in the first line and
|
|
// rebase C2 (the rebased C) onto the resolved conflict. C3 should not have
|
|
// a conflict since it changed an unrelated line.
|
|
let path = RepoPath::from_internal_string("dir/file");
|
|
let mut tx = repo.start_transaction(&settings);
|
|
let tree_a = create_tree(repo, &[(path, "abc\ndef\nghi\n")]);
|
|
let commit_a = tx
|
|
.repo_mut()
|
|
.new_commit(
|
|
&settings,
|
|
vec![repo.store().root_commit_id().clone()],
|
|
tree_a.id(),
|
|
)
|
|
.write()
|
|
.unwrap();
|
|
let tree_b = create_tree(repo, &[(path, "Abc\ndef\nghi\n")]);
|
|
let commit_b = tx
|
|
.repo_mut()
|
|
.new_commit(&settings, vec![commit_a.id().clone()], tree_b.id())
|
|
.write()
|
|
.unwrap();
|
|
let tree_c = create_tree(repo, &[(path, "Abc\ndef\nGhi\n")]);
|
|
let commit_c = tx
|
|
.repo_mut()
|
|
.new_commit(&settings, vec![commit_b.id().clone()], tree_c.id())
|
|
.write()
|
|
.unwrap();
|
|
let tree_d = create_tree(repo, &[(path, "abC\ndef\nghi\n")]);
|
|
let commit_d = tx
|
|
.repo_mut()
|
|
.new_commit(&settings, vec![commit_a.id().clone()], tree_d.id())
|
|
.write()
|
|
.unwrap();
|
|
|
|
let commit_b2 = rebase_commit(
|
|
&settings,
|
|
tx.repo_mut(),
|
|
commit_b,
|
|
vec![commit_d.id().clone()],
|
|
)
|
|
.unwrap();
|
|
let commit_c2 = rebase_commit(
|
|
&settings,
|
|
tx.repo_mut(),
|
|
commit_c,
|
|
vec![commit_b2.id().clone()],
|
|
)
|
|
.unwrap();
|
|
|
|
// Test the setup: Both B and C should have conflicts.
|
|
let tree_b2 = commit_b2.tree().unwrap();
|
|
let tree_c2 = commit_b2.tree().unwrap();
|
|
assert!(!tree_b2.path_value(path).unwrap().is_resolved());
|
|
assert!(!tree_c2.path_value(path).unwrap().is_resolved());
|
|
|
|
// Create the resolved B and rebase C on top.
|
|
let tree_b3 = create_tree(repo, &[(path, "AbC\ndef\nghi\n")]);
|
|
let commit_b3 = tx
|
|
.repo_mut()
|
|
.rewrite_commit(&settings, &commit_b2)
|
|
.set_tree_id(tree_b3.id())
|
|
.write()
|
|
.unwrap();
|
|
let commit_c3 = rebase_commit(
|
|
&settings,
|
|
tx.repo_mut(),
|
|
commit_c2,
|
|
vec![commit_b3.id().clone()],
|
|
)
|
|
.unwrap();
|
|
tx.repo_mut().rebase_descendants(&settings).unwrap();
|
|
let repo = tx.commit("test");
|
|
|
|
// The conflict should now be resolved.
|
|
let tree_c2 = commit_c3.tree().unwrap();
|
|
let resolved_value = tree_c2.path_value(path).unwrap();
|
|
match resolved_value.into_resolved() {
|
|
Ok(Some(TreeValue::File {
|
|
id,
|
|
executable: false,
|
|
})) => {
|
|
assert_eq!(
|
|
testutils::read_file(repo.store(), path, &id),
|
|
b"AbC\ndef\nGhi\n"
|
|
);
|
|
}
|
|
other => {
|
|
panic!("unexpected value: {other:#?}");
|
|
}
|
|
}
|
|
}
|
|
|
|
// TODO: Add tests for simplification of multi-way conflicts. Both the content
|
|
// and the executable bit need testing.
|
|
|
|
#[test]
|
|
fn test_rebase_on_lossy_merge() {
|
|
let settings = testutils::user_settings();
|
|
let test_repo = TestRepo::init();
|
|
let repo = &test_repo.repo;
|
|
|
|
// Test this rebase:
|
|
// D foo=2 D' foo=3
|
|
// |\ |\
|
|
// | C foo=2 | C' foo=3
|
|
// | | => | |
|
|
// B | foo=2 B | foo=2
|
|
// |/ |/
|
|
// A foo=1 A foo=1
|
|
//
|
|
// Commit D effectively discarded a change from "1" to "2", so one
|
|
// reasonable result in D' is "3". That's what the result would be if we
|
|
// didn't have the "A+(A-B)=A" rule. It's also what the result currently
|
|
// is because we don't attempt to resolve the auto-merged parents (if we
|
|
// had, it would have been resolved to just "2" before the rebase and we
|
|
// get a conflict after the rebase).
|
|
let path = RepoPath::from_internal_string("foo");
|
|
let mut tx = repo.start_transaction(&settings);
|
|
let repo_mut = tx.repo_mut();
|
|
let tree_1 = create_tree(repo, &[(path, "1")]);
|
|
let tree_2 = create_tree(repo, &[(path, "2")]);
|
|
let tree_3 = create_tree(repo, &[(path, "3")]);
|
|
let commit_a = repo_mut
|
|
.new_commit(
|
|
&settings,
|
|
vec![repo.store().root_commit_id().clone()],
|
|
tree_1.id(),
|
|
)
|
|
.write()
|
|
.unwrap();
|
|
let commit_b = repo_mut
|
|
.new_commit(&settings, vec![commit_a.id().clone()], tree_2.id())
|
|
.write()
|
|
.unwrap();
|
|
let commit_c = repo_mut
|
|
.new_commit(&settings, vec![commit_a.id().clone()], tree_2.id())
|
|
.write()
|
|
.unwrap();
|
|
let commit_d = repo_mut
|
|
.new_commit(
|
|
&settings,
|
|
vec![commit_b.id().clone(), commit_c.id().clone()],
|
|
tree_2.id(),
|
|
)
|
|
.write()
|
|
.unwrap();
|
|
|
|
let commit_c2 = repo_mut
|
|
.new_commit(&settings, vec![commit_a.id().clone()], tree_3.id())
|
|
.write()
|
|
.unwrap();
|
|
let commit_d2 = rebase_commit(
|
|
&settings,
|
|
repo_mut,
|
|
commit_d,
|
|
vec![commit_b.id().clone(), commit_c2.id().clone()],
|
|
)
|
|
.unwrap();
|
|
|
|
assert_eq!(*commit_d2.tree_id(), tree_3.id());
|
|
}
|