Skip to content

Commit

Permalink
diff: do not emit unified diff for binary files
Browse files Browse the repository at this point in the history
  • Loading branch information
yuja committed Jul 15, 2024
1 parent e3055e5 commit 692c996
Show file tree
Hide file tree
Showing 3 changed files with 57 additions and 23 deletions.
2 changes: 2 additions & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -51,6 +51,8 @@ to [Semantic Versioning](https://semver.org/spec/v2.0.0.html).

### Fixed bugs

* `jj diff --git` no longer shows the contents of binary files.

## [0.19.0] - 2024-07-03

### Breaking changes
Expand Down
59 changes: 36 additions & 23 deletions cli/src/diff_util.rs
Original file line number Diff line number Diff line change
Expand Up @@ -724,7 +724,7 @@ struct GitDiffPart {
/// Octal mode string or `None` if the file is absent.
mode: Option<&'static str>,
hash: String,
content: Vec<u8>,
content: FileContent,
}

fn git_diff_part(
Expand All @@ -734,13 +734,13 @@ fn git_diff_part(
const DUMMY_HASH: &str = "0000000000";
let mode;
let mut hash;
let mut contents: Vec<u8>;
let content;
match value {
MaterializedTreeValue::Absent => {
return Ok(GitDiffPart {
mode: None,
hash: DUMMY_HASH.to_owned(),
content: vec![],
content: FileContent::empty(),
});
}
MaterializedTreeValue::AccessDenied(err) => {
Expand All @@ -756,29 +756,34 @@ fn git_diff_part(
} => {
mode = if executable { "100755" } else { "100644" };
hash = id.hex();
// TODO: use `file_content_for_diff` instead of showing binary
contents = vec![];
reader.read_to_end(&mut contents)?;
content = file_content_for_diff(&mut reader)?;
}
MaterializedTreeValue::Symlink { id, target } => {
mode = "120000";
hash = id.hex();
contents = target.into_bytes();
content = FileContent {
// Unix file paths can't contain null bytes.
is_binary: false,
contents: target.into_bytes(),
};
}
MaterializedTreeValue::GitSubmodule(id) => {
// TODO: What should we actually do here?
mode = "040000";
hash = id.hex();
contents = vec![];
content = FileContent::empty();
}
MaterializedTreeValue::Conflict {
id: _,
contents: conflict_data,
contents,
executable,
} => {
mode = if executable { "100755" } else { "100644" };
hash = DUMMY_HASH.to_owned();
contents = conflict_data
content = FileContent {
is_binary: false, // TODO: are we sure this is never binary?
contents,
};
}
MaterializedTreeValue::Tree(_) => {
panic!("Unexpected tree in diff at path {path:?}");
Expand All @@ -788,7 +793,7 @@ fn git_diff_part(
Ok(GitDiffPart {
mode: Some(mode),
hash,
content: contents,
content,
})
}

Expand Down Expand Up @@ -1058,7 +1063,7 @@ pub fn show_git_diff(
Ok(())
})?;

if left_part.content == right_part.content {
if left_part.content.contents == right_part.content.contents {
continue; // no content hunks
}

Expand All @@ -1070,17 +1075,25 @@ pub fn show_git_diff(
Some(_) => format!("b/{path_string}"),
None => "/dev/null".to_owned(),
};
formatter.with_label("file_header", |formatter| {
writeln!(formatter, "--- {left_path}")?;
writeln!(formatter, "+++ {right_path}")?;
Ok(())
})?;
show_unified_diff_hunks(
formatter,
&left_part.content,
&right_part.content,
num_context_lines,
)?;
if left_part.content.is_binary || right_part.content.is_binary {
// TODO: add option to emit Git binary diff
writeln!(
formatter,
"Binary files {left_path} and {right_path} differ"
)?;
} else {
formatter.with_label("file_header", |formatter| {
writeln!(formatter, "--- {left_path}")?;
writeln!(formatter, "+++ {right_path}")?;
Ok(())
})?;
show_unified_diff_hunks(
formatter,
&left_part.content.contents,
&right_part.content.contents,
num_context_lines,
)?;
}
}
Ok::<(), DiffRenderError>(())
}
Expand Down
19 changes: 19 additions & 0 deletions cli/tests/test_diff_command.rs
Original file line number Diff line number Diff line change
Expand Up @@ -1571,6 +1571,25 @@ fn test_diff_binary() {
(binary)
"###);

let stdout = test_env.jj_cmd_success(&repo_path, &["diff", "--git"]);
insta::assert_snapshot!(stdout, @r###"
diff --git a/file1.png b/file1.png
deleted file mode 100644
index 2b65b23c22..0000000000
Binary files a/file1.png and /dev/null differ
diff --git a/file2.png b/file2.png
index 7f036ce788..3bd1f0e297 100644
Binary files a/file2.png and b/file2.png differ
diff --git a/file3.png b/file3.png
new file mode 100644
index 0000000000..deacfbc286
Binary files /dev/null and b/file3.png differ
diff --git a/file4.png b/file4.png
new file mode 100644
index 0000000000..4227ca4e87
Binary files /dev/null and b/file4.png differ
"###);

let stdout = test_env.jj_cmd_success(&repo_path, &["diff", "--stat"]);
insta::assert_snapshot!(stdout, @r###"
file1.png | 3 ---
Expand Down

0 comments on commit 692c996

Please sign in to comment.