mirror of
https://github.com/martinvonz/jj.git
synced 2025-01-19 19:08:08 +00:00
copies: provide source path mapping by CopyRecords
All for/has_source/target() combinations are added for API consistency.
This commit is contained in:
parent
5e356ffd24
commit
2cffcc9323
2 changed files with 31 additions and 25 deletions
|
@ -396,13 +396,6 @@ pub fn get_copy_records<'a>(
|
|||
Ok(block_on_stream(stream).filter_ok(|record| matcher.matches(&record.target)))
|
||||
}
|
||||
|
||||
fn collect_copied_sources(copy_records: &CopyRecords) -> HashSet<&RepoPath> {
|
||||
copy_records
|
||||
.iter()
|
||||
.map(|record| record.source.as_ref())
|
||||
.collect()
|
||||
}
|
||||
|
||||
#[derive(Clone, Debug, Eq, PartialEq)]
|
||||
pub struct ColorWordsOptions {
|
||||
/// Number of context lines to show.
|
||||
|
@ -919,7 +912,6 @@ pub fn show_file_by_file_diff(
|
|||
std::fs::write(&fs_path, content.contents)?;
|
||||
Ok(fs_path)
|
||||
}
|
||||
let copied_sources = collect_copied_sources(copy_records);
|
||||
|
||||
let temp_dir = new_utf8_temp_dir("jj-diff-")?;
|
||||
let left_wc_dir = temp_dir.path().join("left");
|
||||
|
@ -933,7 +925,7 @@ pub fn show_file_by_file_diff(
|
|||
}) = diff_stream.next().await
|
||||
{
|
||||
let (left_value, right_value) = diff?;
|
||||
if right_value.is_absent() && copied_sources.contains(left_path.as_ref()) {
|
||||
if right_value.is_absent() && copy_records.has_source(&left_path) {
|
||||
continue;
|
||||
}
|
||||
|
||||
|
@ -1272,7 +1264,6 @@ pub fn show_git_diff(
|
|||
) -> Result<(), DiffRenderError> {
|
||||
let tree_diff = from_tree.diff_stream_with_copies(to_tree, matcher, copy_records);
|
||||
let mut diff_stream = materialized_diff_stream(store, tree_diff);
|
||||
let copied_sources = collect_copied_sources(copy_records);
|
||||
|
||||
async {
|
||||
while let Some(MaterializedTreeDiffEntry {
|
||||
|
@ -1289,7 +1280,7 @@ pub fn show_git_diff(
|
|||
let right_part = git_diff_part(&right_path, right_value)?;
|
||||
|
||||
// Skip the "delete" entry when there is a rename.
|
||||
if right_part.mode.is_none() && copied_sources.contains(left_path.as_ref()) {
|
||||
if right_part.mode.is_none() && copy_records.has_source(&left_path) {
|
||||
continue;
|
||||
}
|
||||
|
||||
|
@ -1382,7 +1373,6 @@ pub fn show_diff_summary(
|
|||
copy_records: &CopyRecords,
|
||||
) -> Result<(), DiffRenderError> {
|
||||
let mut tree_diff = from_tree.diff_stream_with_copies(to_tree, matcher, copy_records);
|
||||
let copied_sources = collect_copied_sources(copy_records);
|
||||
|
||||
async {
|
||||
while let Some(CopiesTreeDiffEntry {
|
||||
|
@ -1405,7 +1395,7 @@ pub fn show_diff_summary(
|
|||
(true, true) => writeln!(formatter.labeled("modified"), "M {path}")?,
|
||||
(false, true) => writeln!(formatter.labeled("added"), "A {path}")?,
|
||||
(true, false) => {
|
||||
if !copied_sources.contains(before_path.as_ref()) {
|
||||
if !copy_records.has_source(&before_path) {
|
||||
writeln!(formatter.labeled("removed"), "D {path}")?;
|
||||
}
|
||||
}
|
||||
|
@ -1555,7 +1545,6 @@ pub fn show_types(
|
|||
copy_records: &CopyRecords,
|
||||
) -> Result<(), DiffRenderError> {
|
||||
let mut tree_diff = from_tree.diff_stream_with_copies(to_tree, matcher, copy_records);
|
||||
let copied_sources = collect_copied_sources(copy_records);
|
||||
|
||||
async {
|
||||
while let Some(CopiesTreeDiffEntry {
|
||||
|
@ -1565,7 +1554,7 @@ pub fn show_types(
|
|||
}) = tree_diff.next().await
|
||||
{
|
||||
let (before, after) = diff?;
|
||||
if after.is_absent() && copied_sources.contains(source.as_ref()) {
|
||||
if after.is_absent() && copy_records.has_source(&source) {
|
||||
continue;
|
||||
}
|
||||
writeln!(
|
||||
|
|
|
@ -29,8 +29,9 @@ use crate::repo_path::{RepoPath, RepoPathBuf};
|
|||
#[derive(Default, Debug)]
|
||||
pub struct CopyRecords {
|
||||
records: Vec<CopyRecord>,
|
||||
// Maps from `target` to the index of the target in `records`. Conflicts
|
||||
// are excluded by keeping an out of range value.
|
||||
// Maps from `source` or `target` to the index of the entry in `records`.
|
||||
// Conflicts are excluded by keeping an out of range value.
|
||||
sources: HashMap<RepoPathBuf, usize>,
|
||||
targets: HashMap<RepoPathBuf, usize>,
|
||||
}
|
||||
|
||||
|
@ -43,20 +44,36 @@ impl CopyRecords {
|
|||
) -> BackendResult<()> {
|
||||
for record in copy_records {
|
||||
let r = record?;
|
||||
let value = self
|
||||
.targets
|
||||
.entry(r.target.clone())
|
||||
.or_insert(self.records.len());
|
||||
|
||||
if *value != self.records.len() {
|
||||
self.sources
|
||||
.entry(r.source.clone())
|
||||
// TODO: handle conflicts instead of ignoring both sides.
|
||||
*value = usize::MAX;
|
||||
}
|
||||
.and_modify(|value| *value = usize::MAX)
|
||||
.or_insert(self.records.len());
|
||||
self.targets
|
||||
.entry(r.target.clone())
|
||||
// TODO: handle conflicts instead of ignoring both sides.
|
||||
.and_modify(|value| *value = usize::MAX)
|
||||
.or_insert(self.records.len());
|
||||
self.records.push(r);
|
||||
}
|
||||
Ok(())
|
||||
}
|
||||
|
||||
/// Returns true if there are copy records associated with a source path.
|
||||
pub fn has_source(&self, source: &RepoPath) -> bool {
|
||||
self.sources.contains_key(source)
|
||||
}
|
||||
|
||||
/// Gets any copy record associated with a source path.
|
||||
pub fn for_source(&self, source: &RepoPath) -> Option<&CopyRecord> {
|
||||
self.sources.get(source).and_then(|&i| self.records.get(i))
|
||||
}
|
||||
|
||||
/// Returns true if there are copy records associated with a target path.
|
||||
pub fn has_target(&self, target: &RepoPath) -> bool {
|
||||
self.targets.contains_key(target)
|
||||
}
|
||||
|
||||
/// Gets any copy record associated with a target path.
|
||||
pub fn for_target(&self, target: &RepoPath) -> Option<&CopyRecord> {
|
||||
self.targets.get(target).and_then(|&i| self.records.get(i))
|
||||
|
|
Loading…
Reference in a new issue