From 34b0f875848eae6107ef3a3a22f95c025bc7dd0d Mon Sep 17 00:00:00 2001 From: Matt Kulukundis Date: Fri, 2 Aug 2024 12:48:07 -0400 Subject: [PATCH] copy-tracking: plumb CopyRecordMap through diff method --- cli/src/commands/fix.rs | 3 +- cli/src/commit_templater.rs | 7 +++- cli/src/diff_util.rs | 26 +++++++++---- cli/src/merge_tools/builtin.rs | 2 +- cli/src/merge_tools/diff_working_copies.rs | 2 +- lib/src/default_index/revset_engine.rs | 6 ++- lib/src/local_working_copy.rs | 6 ++- lib/src/merged_tree.rs | 13 ++++++- lib/src/rewrite.rs | 3 +- lib/tests/test_commit_builder.rs | 2 +- lib/tests/test_local_working_copy_sparse.rs | 4 +- lib/tests/test_merged_tree.rs | 41 ++++++++++++--------- 12 files changed, 75 insertions(+), 40 deletions(-) diff --git a/cli/src/commands/fix.rs b/cli/src/commands/fix.rs index 113b7b08f..dc2764cb8 100644 --- a/cli/src/commands/fix.rs +++ b/cli/src/commands/fix.rs @@ -172,7 +172,8 @@ pub(crate) fn cmd_fix( // Also fix any new paths that were changed in this commit. let tree = commit.tree()?; let parent_tree = commit.parent_tree(tx.repo())?; - let mut diff_stream = parent_tree.diff_stream(&tree, &matcher); + let copy_records = Default::default(); + let mut diff_stream = parent_tree.diff_stream(&tree, &matcher, ©_records); async { while let Some(TreeDiffEntry { source: _, // TODO handle copy tracking diff --git a/cli/src/commit_templater.rs b/cli/src/commit_templater.rs index a00656261..c7cb51fd0 100644 --- a/cli/src/commit_templater.rs +++ b/cli/src/commit_templater.rs @@ -19,7 +19,7 @@ use std::io; use std::rc::Rc; use itertools::Itertools as _; -use jj_lib::backend::{BackendResult, ChangeId, CommitId}; +use jj_lib::backend::{BackendResult, ChangeId, CommitId, CopyRecords}; use jj_lib::commit::Commit; use jj_lib::extensions_map::ExtensionsMap; use jj_lib::fileset::{self, FilesetExpression}; @@ -1278,6 +1278,7 @@ pub struct TreeDiff { from_tree: MergedTree, to_tree: MergedTree, matcher: Rc, + copy_records: CopyRecords, } impl TreeDiff { @@ -1290,11 +1291,13 @@ impl TreeDiff { from_tree: commit.parent_tree(repo)?, to_tree: commit.tree()?, matcher, + copy_records: Default::default(), }) } fn diff_stream(&self) -> TreeDiffStream<'_> { - self.from_tree.diff_stream(&self.to_tree, &*self.matcher) + self.from_tree + .diff_stream(&self.to_tree, &*self.matcher, &self.copy_records) } fn into_formatted(self, show: F) -> TreeDiffFormatted diff --git a/cli/src/diff_util.rs b/cli/src/diff_util.rs index b52cde1c4..137954735 100644 --- a/cli/src/diff_util.rs +++ b/cli/src/diff_util.rs @@ -240,6 +240,7 @@ impl<'a> DiffRenderer<'a> { } /// Generates diff between `from_tree` and `to_tree`. + #[allow(clippy::too_many_arguments)] pub fn show_diff( &self, ui: &Ui, // TODO: remove Ui dependency if possible @@ -263,6 +264,7 @@ impl<'a> DiffRenderer<'a> { }) } + #[allow(clippy::too_many_arguments)] fn show_diff_inner( &self, ui: &Ui, @@ -278,33 +280,41 @@ impl<'a> DiffRenderer<'a> { for format in &self.formats { match format { DiffFormat::Summary => { - let tree_diff = from_tree.diff_stream(to_tree, matcher); + let no_copy_tracking = Default::default(); + let tree_diff = from_tree.diff_stream(to_tree, matcher, &no_copy_tracking); show_diff_summary(formatter, tree_diff, path_converter)?; } DiffFormat::Stat => { - let tree_diff = from_tree.diff_stream(to_tree, matcher); + let no_copy_tracking = Default::default(); + let tree_diff = from_tree.diff_stream(to_tree, matcher, &no_copy_tracking); show_diff_stat(formatter, store, tree_diff, path_converter, width)?; } DiffFormat::Types => { - let tree_diff = from_tree.diff_stream(to_tree, matcher); + let no_copy_tracking = Default::default(); + let tree_diff = from_tree.diff_stream(to_tree, matcher, &no_copy_tracking); show_types(formatter, tree_diff, path_converter)?; } DiffFormat::NameOnly => { - let tree_diff = from_tree.diff_stream(to_tree, matcher); + let no_copy_tracking = Default::default(); + let tree_diff = from_tree.diff_stream(to_tree, matcher, &no_copy_tracking); show_names(formatter, tree_diff, path_converter)?; } DiffFormat::Git { context } => { - let tree_diff = from_tree.diff_stream(to_tree, matcher); + let no_copy_tracking = Default::default(); + let tree_diff = from_tree.diff_stream(to_tree, matcher, &no_copy_tracking); show_git_diff(formatter, store, tree_diff, *context)?; } DiffFormat::ColorWords { context } => { - let tree_diff = from_tree.diff_stream(to_tree, matcher); + let no_copy_tracking = Default::default(); + let tree_diff = from_tree.diff_stream(to_tree, matcher, &no_copy_tracking); show_color_words_diff(formatter, store, tree_diff, path_converter, *context)?; } DiffFormat::Tool(tool) => { match tool.diff_invocation_mode { DiffToolMode::FileByFile => { - let tree_diff = from_tree.diff_stream(to_tree, matcher); + let no_copy_tracking = Default::default(); + let tree_diff = + from_tree.diff_stream(to_tree, matcher, &no_copy_tracking); show_file_by_file_diff( ui, formatter, @@ -1052,7 +1062,7 @@ pub fn show_git_diff( { let path_string = path.as_internal_file_string(); let (left_value, right_value) = diff?; - let left_part = git_digf_part(&path, left_value)?; + let left_part = git_diff_part(&path, left_value)?; let right_part = git_diff_part(&path, right_value)?; formatter.with_label("file_header", |formatter| { writeln!(formatter, "diff --git a/{path_string} b/{path_string}")?; diff --git a/cli/src/merge_tools/builtin.rs b/cli/src/merge_tools/builtin.rs index 70dc01108..a9bfbcd42 100644 --- a/cli/src/merge_tools/builtin.rs +++ b/cli/src/merge_tools/builtin.rs @@ -495,7 +495,7 @@ pub fn edit_diff_builtin( ) -> Result { let store = left_tree.store().clone(); let changed_files: Vec<_> = left_tree - .diff_stream(right_tree, matcher) + .diff_stream(right_tree, matcher, &Default::default()) .map( |TreeDiffEntry { source: _, // TODO handle copy tracking diff --git a/cli/src/merge_tools/diff_working_copies.rs b/cli/src/merge_tools/diff_working_copies.rs index 1e6d43c38..2c67aa5a2 100644 --- a/cli/src/merge_tools/diff_working_copies.rs +++ b/cli/src/merge_tools/diff_working_copies.rs @@ -131,7 +131,7 @@ pub(crate) fn check_out_trees( output_is: Option, ) -> Result { let changed_files: Vec<_> = left_tree - .diff_stream(right_tree, matcher) + .diff_stream(right_tree, matcher, &Default::default()) .map(|TreeDiffEntry { target, .. }| target) .collect() .block_on(); diff --git a/lib/src/default_index/revset_engine.rs b/lib/src/default_index/revset_engine.rs index 6eebdc0e2..5fd14e64c 100644 --- a/lib/src/default_index/revset_engine.rs +++ b/lib/src/default_index/revset_engine.rs @@ -1148,7 +1148,8 @@ fn has_diff_from_parent( let from_tree = rewrite::merge_commit_trees_no_resolve_without_repo(store, &index, &parents)?.resolve()?; let to_tree = commit.tree()?; - let mut tree_diff = from_tree.diff_stream(&to_tree, matcher); + let copy_records = Default::default(); + let mut tree_diff = from_tree.diff_stream(&to_tree, matcher, ©_records); async { match tree_diff.next().await { Some(TreeDiffEntry { value: Ok(_), .. }) => Ok(true), @@ -1168,11 +1169,12 @@ fn matches_diff_from_parent( text_pattern: &StringPattern, files_matcher: &dyn Matcher, ) -> BackendResult { + let copy_records = Default::default(); let parents: Vec<_> = commit.parents().try_collect()?; let from_tree = rewrite::merge_commit_trees_no_resolve_without_repo(store, &index, &parents)?.resolve()?; let to_tree = commit.tree()?; - let tree_diff = from_tree.diff_stream(&to_tree, files_matcher); + let tree_diff = from_tree.diff_stream(&to_tree, files_matcher, ©_records); // Conflicts are compared in materialized form. Alternatively, conflict // pairs can be compared one by one. #4062 let mut diff_stream = materialized_diff_stream(store, tree_diff); diff --git a/lib/src/local_working_copy.rs b/lib/src/local_working_copy.rs index d81831566..9c5ee09c4 100644 --- a/lib/src/local_working_copy.rs +++ b/lib/src/local_working_copy.rs @@ -1340,6 +1340,7 @@ impl TreeState { new_tree: &MergedTree, matcher: &dyn Matcher, ) -> Result { + let copy_records = Default::default(); // TODO: maybe it's better not include the skipped counts in the "intended" // counts let mut stats = CheckoutStats { @@ -1352,7 +1353,7 @@ impl TreeState { let mut deleted_files = HashSet::new(); let mut diff_stream = Box::pin( old_tree - .diff_stream(new_tree, matcher) + .diff_stream(new_tree, matcher, ©_records) .map( |TreeDiffEntry { source: _, // TODO handle copy tracking @@ -1451,9 +1452,10 @@ impl TreeState { })?; let matcher = self.sparse_matcher(); + let copy_records = Default::default(); let mut changed_file_states = Vec::new(); let mut deleted_files = HashSet::new(); - let mut diff_stream = old_tree.diff_stream(new_tree, matcher.as_ref()); + let mut diff_stream = old_tree.diff_stream(new_tree, matcher.as_ref(), ©_records); while let Some(TreeDiffEntry { source: _, // TODO handle copy tracking target: path, diff --git a/lib/src/merged_tree.rs b/lib/src/merged_tree.rs index 8643453f1..e6ed59ec1 100644 --- a/lib/src/merged_tree.rs +++ b/lib/src/merged_tree.rs @@ -29,7 +29,7 @@ use futures::{Stream, TryStreamExt}; use itertools::{EitherOrBoth, Itertools}; use crate::backend; -use crate::backend::{BackendResult, MergedTreeId, TreeId, TreeValue}; +use crate::backend::{BackendResult, CopyRecords, MergedTreeId, TreeId, TreeValue}; use crate::matchers::{EverythingMatcher, Matcher}; use crate::merge::{Merge, MergeBuilder, MergedTreeValue}; use crate::repo_path::{RepoPath, RepoPathBuf, RepoPathComponent}; @@ -283,6 +283,7 @@ impl MergedTree { &self, other: &MergedTree, matcher: &'matcher dyn Matcher, + copy_records: &'matcher CopyRecords, ) -> TreeDiffStream<'matcher> { let concurrency = self.store().concurrency(); if concurrency <= 1 { @@ -290,12 +291,14 @@ impl MergedTree { &self.trees, &other.trees, matcher, + copy_records, ))) } else { Box::pin(TreeDiffStreamImpl::new( self.clone(), other.clone(), matcher, + copy_records, concurrency, )) } @@ -627,7 +630,12 @@ enum TreeDiffItem { impl<'matcher> TreeDiffIterator<'matcher> { /// Creates a iterator over the differences between two trees. Generally /// prefer `MergedTree::diff()` of calling this directly. - pub fn new(trees1: &Merge, trees2: &Merge, matcher: &'matcher dyn Matcher) -> Self { + pub fn new( + trees1: &Merge, + trees2: &Merge, + matcher: &'matcher dyn Matcher, + _copy_records: &'matcher CopyRecords, + ) -> Self { assert!(Arc::ptr_eq(trees1.first().store(), trees2.first().store())); let root_dir = RepoPath::root(); let mut stack = Vec::new(); @@ -857,6 +865,7 @@ impl<'matcher> TreeDiffStreamImpl<'matcher> { tree1: MergedTree, tree2: MergedTree, matcher: &'matcher dyn Matcher, + _copy_records: &'matcher CopyRecords, max_concurrent_reads: usize, ) -> Self { let mut stream = Self { diff --git a/lib/src/rewrite.rs b/lib/src/rewrite.rs index 7ccc59a04..5045d2c58 100644 --- a/lib/src/rewrite.rs +++ b/lib/src/rewrite.rs @@ -86,8 +86,9 @@ pub fn restore_tree( // TODO: We should be able to not traverse deeper in the diff if the matcher // matches an entire subtree. let mut tree_builder = MergedTreeBuilder::new(destination.id().clone()); + let copy_records = Default::default(); async { - let mut diff_stream = source.diff_stream(destination, matcher); + let mut diff_stream = source.diff_stream(destination, matcher, ©_records); while let Some(TreeDiffEntry { source: _, // TODO handle copy tracking target: repo_path, diff --git a/lib/tests/test_commit_builder.rs b/lib/tests/test_commit_builder.rs index b3f345925..6ca01d96a 100644 --- a/lib/tests/test_commit_builder.rs +++ b/lib/tests/test_commit_builder.rs @@ -26,7 +26,7 @@ use testutils::{assert_rebased_onto, create_tree, CommitGraphBuilder, TestRepo, fn diff_paths(from_tree: &MergedTree, to_tree: &MergedTree) -> Vec { from_tree - .diff_stream(to_tree, &EverythingMatcher) + .diff_stream(to_tree, &EverythingMatcher, &Default::default()) .map(|diff| { let _ = diff.value.unwrap(); diff.target diff --git a/lib/tests/test_local_working_copy_sparse.rs b/lib/tests/test_local_working_copy_sparse.rs index 3b9b50956..06d4ccfe4 100644 --- a/lib/tests/test_local_working_copy_sparse.rs +++ b/lib/tests/test_local_working_copy_sparse.rs @@ -197,7 +197,7 @@ fn test_sparse_commit() { // tree. let modified_tree = test_workspace.snapshot().unwrap(); let diff: Vec<_> = tree - .diff_stream(&modified_tree, &EverythingMatcher) + .diff_stream(&modified_tree, &EverythingMatcher, &Default::default()) .collect() .block_on(); assert_eq!(diff.len(), 1); @@ -219,7 +219,7 @@ fn test_sparse_commit() { // updated in the tree. let modified_tree = test_workspace.snapshot().unwrap(); let diff: Vec<_> = tree - .diff_stream(&modified_tree, &EverythingMatcher) + .diff_stream(&modified_tree, &EverythingMatcher, &Default::default()) .collect() .block_on(); assert_eq!(diff.len(), 2); diff --git a/lib/tests/test_merged_tree.rs b/lib/tests/test_merged_tree.rs index 3617cc1ed..b8196c33c 100644 --- a/lib/tests/test_merged_tree.rs +++ b/lib/tests/test_merged_tree.rs @@ -40,15 +40,22 @@ fn diff_entry_tuple(diff: TreeDiffEntry) -> (RepoPathBuf, (MergedTreeValue, Merg } fn diff_stream_equals_iter(tree1: &MergedTree, tree2: &MergedTree, matcher: &dyn Matcher) { - let iter_diff: Vec<_> = TreeDiffIterator::new(tree1.as_merge(), tree2.as_merge(), matcher) - .map(diff_entry_tuple) - .collect(); - let max_concurrent_reads = 10; - let stream_diff: Vec<_> = - TreeDiffStreamImpl::new(tree1.clone(), tree2.clone(), matcher, max_concurrent_reads) + let copy_records = Default::default(); + let iter_diff: Vec<_> = + TreeDiffIterator::new(tree1.as_merge(), tree2.as_merge(), matcher, ©_records) .map(diff_entry_tuple) - .collect() - .block_on(); + .collect(); + let max_concurrent_reads = 10; + let stream_diff: Vec<_> = TreeDiffStreamImpl::new( + tree1.clone(), + tree2.clone(), + matcher, + ©_records, + max_concurrent_reads, + ) + .map(diff_entry_tuple) + .collect() + .block_on(); assert_eq!(stream_diff, iter_diff); } @@ -752,7 +759,7 @@ fn test_diff_resolved() { let after_merged = MergedTree::new(Merge::resolved(after.clone())); let diff: Vec<_> = before_merged - .diff_stream(&after_merged, &EverythingMatcher) + .diff_stream(&after_merged, &EverythingMatcher, &Default::default()) .map(diff_entry_tuple) .collect() .block_on(); @@ -862,7 +869,7 @@ fn test_diff_conflicted() { // Test the forwards diff let actual_diff: Vec<_> = left_merged - .diff_stream(&right_merged, &EverythingMatcher) + .diff_stream(&right_merged, &EverythingMatcher, &Default::default()) .map(diff_entry_tuple) .collect() .block_on(); @@ -882,7 +889,7 @@ fn test_diff_conflicted() { diff_stream_equals_iter(&left_merged, &right_merged, &EverythingMatcher); // Test the reverse diff let actual_diff: Vec<_> = right_merged - .diff_stream(&left_merged, &EverythingMatcher) + .diff_stream(&left_merged, &EverythingMatcher, &Default::default()) .map(diff_entry_tuple) .collect() .block_on(); @@ -1000,7 +1007,7 @@ fn test_diff_dir_file() { // Test the forwards diff { let actual_diff: Vec<_> = left_merged - .diff_stream(&right_merged, &EverythingMatcher) + .diff_stream(&right_merged, &EverythingMatcher, &Default::default()) .map(diff_entry_tuple) .collect() .block_on(); @@ -1045,7 +1052,7 @@ fn test_diff_dir_file() { // Test the reverse diff { let actual_diff: Vec<_> = right_merged - .diff_stream(&left_merged, &EverythingMatcher) + .diff_stream(&left_merged, &EverythingMatcher, &Default::default()) .map(diff_entry_tuple) .collect() .block_on(); @@ -1091,7 +1098,7 @@ fn test_diff_dir_file() { { let matcher = FilesMatcher::new([&path1]); let actual_diff: Vec<_> = left_merged - .diff_stream(&right_merged, &matcher) + .diff_stream(&right_merged, &matcher, &Default::default()) .map(diff_entry_tuple) .collect() .block_on(); @@ -1107,7 +1114,7 @@ fn test_diff_dir_file() { { let matcher = FilesMatcher::new([path1.join(file)]); let actual_diff: Vec<_> = left_merged - .diff_stream(&right_merged, &matcher) + .diff_stream(&right_merged, &matcher, &Default::default()) .map(diff_entry_tuple) .collect() .block_on(); @@ -1126,7 +1133,7 @@ fn test_diff_dir_file() { { let matcher = PrefixMatcher::new([&path1]); let actual_diff: Vec<_> = left_merged - .diff_stream(&right_merged, &matcher) + .diff_stream(&right_merged, &matcher, &Default::default()) .map(diff_entry_tuple) .collect() .block_on(); @@ -1148,7 +1155,7 @@ fn test_diff_dir_file() { { let matcher = FilesMatcher::new([&path6]); let actual_diff: Vec<_> = left_merged - .diff_stream(&right_merged, &matcher) + .diff_stream(&right_merged, &matcher, &Default::default()) .map(diff_entry_tuple) .collect() .block_on();