mirror of
https://github.com/martinvonz/jj.git
synced 2025-01-24 21:13:47 +00:00
rewrite: fix duplicated commits to be rebased onto destination
I believe this was an oversight. "jj duplicate" should duplicate commits (= patches), not trees. This patch adds a separate test file because test_rewrite.rs is pretty big, and we'll probably want to migrate CLI tests to jj-lib.
This commit is contained in:
parent
a6c18e8353
commit
5cc0bd0950
4 changed files with 577 additions and 399 deletions
File diff suppressed because it is too large
Load diff
|
@ -878,10 +878,9 @@ pub fn duplicate_commits(
|
||||||
})
|
})
|
||||||
.collect()
|
.collect()
|
||||||
};
|
};
|
||||||
let new_commit = mut_repo
|
let new_commit = CommitRewriter::new(mut_repo, original_commit, new_parent_ids)
|
||||||
.rewrite_commit(settings, &original_commit)
|
.rebase(settings)?
|
||||||
.generate_new_change_id()
|
.generate_new_change_id()
|
||||||
.set_parents(new_parent_ids)
|
|
||||||
.write()?;
|
.write()?;
|
||||||
duplicated_old_to_new.insert(original_commit_id.clone(), new_commit);
|
duplicated_old_to_new.insert(original_commit_id.clone(), new_commit);
|
||||||
}
|
}
|
||||||
|
|
|
@ -29,6 +29,7 @@ mod test_operations;
|
||||||
mod test_refs;
|
mod test_refs;
|
||||||
mod test_revset;
|
mod test_revset;
|
||||||
mod test_rewrite;
|
mod test_rewrite;
|
||||||
|
mod test_rewrite_duplicate;
|
||||||
mod test_rewrite_transform;
|
mod test_rewrite_transform;
|
||||||
mod test_signing;
|
mod test_signing;
|
||||||
mod test_ssh_signing;
|
mod test_ssh_signing;
|
||||||
|
|
182
lib/tests/test_rewrite_duplicate.rs
Normal file
182
lib/tests/test_rewrite_duplicate.rs
Normal file
|
@ -0,0 +1,182 @@
|
||||||
|
// Copyright 2024 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 as _;
|
||||||
|
use jj_lib::backend::CommitId;
|
||||||
|
use jj_lib::repo::Repo;
|
||||||
|
use jj_lib::repo_path::RepoPath;
|
||||||
|
use jj_lib::rewrite::duplicate_commits;
|
||||||
|
use jj_lib::transaction::Transaction;
|
||||||
|
use testutils::create_tree;
|
||||||
|
use testutils::TestRepo;
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_duplicate_linear_contents() {
|
||||||
|
let settings = testutils::user_settings();
|
||||||
|
let test_repo = TestRepo::init();
|
||||||
|
let repo = &test_repo.repo;
|
||||||
|
|
||||||
|
let path_1 = RepoPath::from_internal_string("file1");
|
||||||
|
let path_2 = RepoPath::from_internal_string("file2");
|
||||||
|
let empty_tree_id = repo.store().empty_merged_tree_id();
|
||||||
|
let tree_1 = create_tree(repo, &[(path_1, "content1")]);
|
||||||
|
let tree_2 = create_tree(repo, &[(path_2, "content2")]);
|
||||||
|
let tree_1_2 = create_tree(repo, &[(path_1, "content1"), (path_2, "content2")]);
|
||||||
|
|
||||||
|
// E [=file2]
|
||||||
|
// D [-file1, =file2]
|
||||||
|
// C [=file1, +file2]
|
||||||
|
// B [+file1]
|
||||||
|
// A []
|
||||||
|
let mut tx = repo.start_transaction(&settings);
|
||||||
|
let commit_a = tx
|
||||||
|
.repo_mut()
|
||||||
|
.new_commit(
|
||||||
|
&settings,
|
||||||
|
vec![repo.store().root_commit_id().clone()],
|
||||||
|
empty_tree_id.clone(),
|
||||||
|
)
|
||||||
|
.write()
|
||||||
|
.unwrap();
|
||||||
|
let commit_b = tx
|
||||||
|
.repo_mut()
|
||||||
|
.new_commit(&settings, vec![commit_a.id().clone()], tree_1.id())
|
||||||
|
.write()
|
||||||
|
.unwrap();
|
||||||
|
let commit_c = tx
|
||||||
|
.repo_mut()
|
||||||
|
.new_commit(&settings, vec![commit_b.id().clone()], tree_1_2.id())
|
||||||
|
.write()
|
||||||
|
.unwrap();
|
||||||
|
let commit_d = tx
|
||||||
|
.repo_mut()
|
||||||
|
.new_commit(&settings, vec![commit_c.id().clone()], tree_2.id())
|
||||||
|
.write()
|
||||||
|
.unwrap();
|
||||||
|
let commit_e = tx
|
||||||
|
.repo_mut()
|
||||||
|
.new_commit(&settings, vec![commit_d.id().clone()], tree_2.id())
|
||||||
|
.write()
|
||||||
|
.unwrap();
|
||||||
|
let repo = tx.commit("test").unwrap();
|
||||||
|
|
||||||
|
let duplicate_in_between = |tx: &mut Transaction,
|
||||||
|
target_commits: &[&CommitId],
|
||||||
|
parent_commit_ids: &[&CommitId],
|
||||||
|
children_commit_ids: &[&CommitId]| {
|
||||||
|
duplicate_commits(
|
||||||
|
&settings,
|
||||||
|
tx.repo_mut(),
|
||||||
|
&target_commits.iter().copied().cloned().collect_vec(),
|
||||||
|
&parent_commit_ids.iter().copied().cloned().collect_vec(),
|
||||||
|
&children_commit_ids.iter().copied().cloned().collect_vec(),
|
||||||
|
)
|
||||||
|
.unwrap()
|
||||||
|
};
|
||||||
|
let duplicate_onto =
|
||||||
|
|tx: &mut Transaction, target_commits: &[&CommitId], parent_commit_ids: &[&CommitId]| {
|
||||||
|
duplicate_in_between(tx, target_commits, parent_commit_ids, &[])
|
||||||
|
};
|
||||||
|
|
||||||
|
// Duplicate empty commit onto empty ancestor tree
|
||||||
|
let mut tx = repo.start_transaction(&settings);
|
||||||
|
let stats = duplicate_onto(&mut tx, &[commit_e.id()], &[commit_a.id()]);
|
||||||
|
assert_eq!(
|
||||||
|
stats.duplicated_commits[commit_e.id()].tree_id(),
|
||||||
|
&empty_tree_id
|
||||||
|
);
|
||||||
|
|
||||||
|
// Duplicate empty commit onto non-empty ancestor tree
|
||||||
|
let mut tx = repo.start_transaction(&settings);
|
||||||
|
let stats = duplicate_onto(&mut tx, &[commit_e.id()], &[commit_b.id()]);
|
||||||
|
assert_eq!(
|
||||||
|
stats.duplicated_commits[commit_e.id()].tree_id(),
|
||||||
|
&tree_1.id()
|
||||||
|
);
|
||||||
|
|
||||||
|
// Duplicate non-empty commit onto empty ancestor tree
|
||||||
|
let mut tx = repo.start_transaction(&settings);
|
||||||
|
let stats = duplicate_onto(&mut tx, &[commit_c.id()], &[commit_a.id()]);
|
||||||
|
assert_eq!(
|
||||||
|
stats.duplicated_commits[commit_c.id()].tree_id(),
|
||||||
|
&tree_2.id()
|
||||||
|
);
|
||||||
|
|
||||||
|
// Duplicate non-empty commit onto non-empty ancestor tree
|
||||||
|
let mut tx = repo.start_transaction(&settings);
|
||||||
|
let stats = duplicate_onto(&mut tx, &[commit_d.id()], &[commit_b.id()]);
|
||||||
|
assert_eq!(
|
||||||
|
stats.duplicated_commits[commit_d.id()].tree_id(),
|
||||||
|
&empty_tree_id
|
||||||
|
);
|
||||||
|
|
||||||
|
// Duplicate non-empty commit onto non-empty descendant tree
|
||||||
|
let mut tx = repo.start_transaction(&settings);
|
||||||
|
let stats = duplicate_onto(&mut tx, &[commit_b.id()], &[commit_d.id()]);
|
||||||
|
assert_eq!(
|
||||||
|
stats.duplicated_commits[commit_b.id()].tree_id(),
|
||||||
|
&tree_1_2.id()
|
||||||
|
);
|
||||||
|
|
||||||
|
// Duplicate multiple contiguous commits
|
||||||
|
let mut tx = repo.start_transaction(&settings);
|
||||||
|
let stats = duplicate_onto(&mut tx, &[commit_e.id(), commit_d.id()], &[commit_b.id()]);
|
||||||
|
assert_eq!(
|
||||||
|
stats.duplicated_commits[commit_d.id()].tree_id(),
|
||||||
|
&empty_tree_id
|
||||||
|
);
|
||||||
|
assert_eq!(
|
||||||
|
stats.duplicated_commits[commit_e.id()].tree_id(),
|
||||||
|
&empty_tree_id
|
||||||
|
);
|
||||||
|
|
||||||
|
// Duplicate multiple non-contiguous commits
|
||||||
|
let mut tx = repo.start_transaction(&settings);
|
||||||
|
let stats = duplicate_onto(&mut tx, &[commit_e.id(), commit_c.id()], &[commit_a.id()]);
|
||||||
|
assert_eq!(
|
||||||
|
stats.duplicated_commits[commit_c.id()].tree_id(),
|
||||||
|
&tree_2.id()
|
||||||
|
);
|
||||||
|
assert_eq!(
|
||||||
|
stats.duplicated_commits[commit_e.id()].tree_id(),
|
||||||
|
&tree_2.id()
|
||||||
|
);
|
||||||
|
|
||||||
|
// Duplicate onto multiple parents
|
||||||
|
let mut tx = repo.start_transaction(&settings);
|
||||||
|
let stats = duplicate_onto(&mut tx, &[commit_d.id()], &[commit_c.id(), commit_b.id()]);
|
||||||
|
assert_eq!(
|
||||||
|
stats.duplicated_commits[commit_d.id()].tree_id(),
|
||||||
|
&tree_2.id()
|
||||||
|
);
|
||||||
|
|
||||||
|
// Insert duplicated commit
|
||||||
|
let mut tx = repo.start_transaction(&settings);
|
||||||
|
let stats = duplicate_in_between(
|
||||||
|
&mut tx,
|
||||||
|
&[commit_b.id()],
|
||||||
|
&[commit_d.id()],
|
||||||
|
&[commit_e.id()],
|
||||||
|
);
|
||||||
|
assert_eq!(
|
||||||
|
stats.duplicated_commits[commit_b.id()].tree_id(),
|
||||||
|
&tree_1_2.id()
|
||||||
|
);
|
||||||
|
let (head_id,) = tx.repo().view().heads().iter().collect_tuple().unwrap();
|
||||||
|
assert_ne!(head_id, commit_e.id());
|
||||||
|
assert_eq!(
|
||||||
|
tx.repo().store().get_commit(head_id).unwrap().tree_id(),
|
||||||
|
&tree_1_2.id()
|
||||||
|
);
|
||||||
|
}
|
Loading…
Reference in a new issue