* git diff --exit-code returns 0 when binary files differ
@ 2024-09-21 4:26 Kohei Shibata
2024-09-21 15:09 ` [PATCH] diff: report modified binary files as changes in builtin_diff() René Scharfe
0 siblings, 1 reply; 4+ messages in thread
From: Kohei Shibata @ 2024-09-21 4:26 UTC (permalink / raw)
To: git
I've encountered an issue with `git diff --exit-code` where it returns
0 for binary files that have actual changes.
> What did you do before the bug happened? (Steps to reproduce your issue)
1. Initialize a new git repository:
```
git init
```
2. Create a binary file and commit it:
```
echo '*.bin binary' > .gitattributes
dd if=/dev/urandom of=a.bin bs=32 count=1
git add .
git commit -m 'commit'
```
3. Modify the binary file:
```
echo a > a.bin
git diff --exit-code # says "Binary files a/a.bin and b/a.bin differ"
echo $? # returns 0
```
> What did you expect to happen? (Expected behavior)
`git diff --exit-code` should exit with 1
> What happened instead? (Actual behavior)
`git diff --exit-code` returns 0 even when the binary file is modified.
> Anything else you want to add:
I could not find the exact condition to change exit code. In some
cases, depending on the content of the file, `git diff --exit-code`
does return 1 as expected.
I don't use an external diff tool.
[System Info]
git version:
git version 2.46.1
cpu: x86_64
no commit associated with this build
sizeof-long: 8
sizeof-size_t: 8
shell-path: /bin/sh
libcurl: 7.68.0
zlib: 1.2.11
uname: Linux 5.15.153.1-microsoft-standard-WSL2 #1 SMP Fri Mar 29
23:14:13 UTC 2024 x86_64
compiler info: gnuc: 9.4
libc info: glibc: 2.31
$SHELL (typically, interactive shell): /bin/bash
[Enabled Hooks]
Best regards,
Kohei
^ permalink raw reply [flat|nested] 4+ messages in thread
* [PATCH] diff: report modified binary files as changes in builtin_diff()
2024-09-21 4:26 git diff --exit-code returns 0 when binary files differ Kohei Shibata
@ 2024-09-21 15:09 ` René Scharfe
2024-09-25 21:24 ` Thomas Braun
0 siblings, 1 reply; 4+ messages in thread
From: René Scharfe @ 2024-09-21 15:09 UTC (permalink / raw)
To: Kohei Shibata, git
The diff machinery has two ways to detect changes to set the exit code:
Just comparing hashes and comparing blob contents. The latter is needed
if certain changes have to be ignored, e.g. with --ignore-space-change
or --ignore-matching-lines. It's enabled by the diff_options flag
diff_from_contents.
The code for handling binary files added by 1aaf69e669 (diff: shortcut
for diff'ing two binary SHA-1 objects, 2014-08-16) always uses a quick
hash-only comparison, even if the slow way is taken. We need it to
report a hash difference as a change for the purpose of setting the
exit code, though, but it never did. Fix that.
d7b97b7185 (diff: let external diffs report that changes are
uninteresting, 2024-06-09) set diff_from_contents if external diff
programs are allowed. This is the default e.g. for git diff, and so
that change exposed the inconsistency much more widely.
Reported-by: Kohei Shibata <shiba200712@gmail.com>
Signed-off-by: René Scharfe <l.s.r@web.de>
---
Thank you for the report!
diff.c | 1 +
t/t4017-diff-retval.sh | 8 ++++++++
2 files changed, 9 insertions(+)
diff --git a/diff.c b/diff.c
index 3be927b073..84a6bb0868 100644
--- a/diff.c
+++ b/diff.c
@@ -3675,6 +3675,7 @@ static void builtin_diff(const char *name_a,
emit_diff_symbol(o, DIFF_SYMBOL_BINARY_FILES,
sb.buf, sb.len, 0);
strbuf_release(&sb);
+ o->found_changes = 1;
goto free_ab_and_return;
}
if (fill_mmfile(o->repo, &mf1, one) < 0 ||
diff --git a/t/t4017-diff-retval.sh b/t/t4017-diff-retval.sh
index d644310e22..1cea73ef5a 100755
--- a/t/t4017-diff-retval.sh
+++ b/t/t4017-diff-retval.sh
@@ -145,6 +145,14 @@ test_expect_success 'option errors are not confused by --exit-code' '
for option in --exit-code --quiet
do
+ test_expect_success "git diff $option returns 1 for changed binary file" "
+ test_when_finished 'rm -f .gitattributes' &&
+ git reset --hard &&
+ echo a binary >.gitattributes &&
+ echo 2 >>a &&
+ test_expect_code 1 git diff $option
+ "
+
test_expect_success "git diff $option returns 1 for copied file" "
git reset --hard &&
cp a copy &&
--
2.46.0
^ permalink raw reply related [flat|nested] 4+ messages in thread
* Re: [PATCH] diff: report modified binary files as changes in builtin_diff()
2024-09-21 15:09 ` [PATCH] diff: report modified binary files as changes in builtin_diff() René Scharfe
@ 2024-09-25 21:24 ` Thomas Braun
2024-09-25 21:52 ` Junio C Hamano
0 siblings, 1 reply; 4+ messages in thread
From: Thomas Braun @ 2024-09-25 21:24 UTC (permalink / raw)
To: René Scharfe, Kohei Shibata, git
Am 21.09.2024 um 17:09 schrieb René Scharfe:
Hi René,
> diff --git a/diff.c b/diff.c
> index 3be927b073..84a6bb0868 100644
> --- a/diff.c
> +++ b/diff.c
> @@ -3675,6 +3675,7 @@ static void builtin_diff(const char *name_a,
> emit_diff_symbol(o, DIFF_SYMBOL_BINARY_FILES,
> sb.buf, sb.len, 0);
> strbuf_release(&sb);
> + o->found_changes = 1;
> goto free_ab_and_return;
> }
> if (fill_mmfile(o->repo, &mf1, one) < 0 ||
I poked at the same issue in parallel and had the same fix, but ...
> diff --git a/t/t4017-diff-retval.sh b/t/t4017-diff-retval.sh
> index d644310e22..1cea73ef5a 100755
> --- a/t/t4017-diff-retval.sh
> +++ b/t/t4017-diff-retval.sh
> @@ -145,6 +145,14 @@ test_expect_success 'option errors are not confused by --exit-code' '
>
> for option in --exit-code --quiet
> do
> + test_expect_success "git diff $option returns 1 for changed binary file" "
> + test_when_finished 'rm -f .gitattributes' &&
> + git reset --hard &&
> + echo a binary >.gitattributes &&
> + echo 2 >>a &&
> + test_expect_code 1 git diff $option
> + "
> +
> test_expect_success "git diff $option returns 1 for copied file" "
> git reset --hard &&
> cp a copy &&
your test is nicer.
The patch works here locally.
For what it's worth:
Reviewed-by: Thomas Braun <thomas.braun@virtuell-zuhause.de>
Thomas
^ permalink raw reply [flat|nested] 4+ messages in thread
* Re: [PATCH] diff: report modified binary files as changes in builtin_diff()
2024-09-25 21:24 ` Thomas Braun
@ 2024-09-25 21:52 ` Junio C Hamano
0 siblings, 0 replies; 4+ messages in thread
From: Junio C Hamano @ 2024-09-25 21:52 UTC (permalink / raw)
To: Thomas Braun; +Cc: René Scharfe, Kohei Shibata, git
Thomas Braun <thomas.braun@virtuell-zuhause.de> writes:
> Am 21.09.2024 um 17:09 schrieb René Scharfe:
>
> Hi René,
>
>> diff --git a/diff.c b/diff.c
>> index 3be927b073..84a6bb0868 100644
>> --- a/diff.c
>> +++ b/diff.c
>> @@ -3675,6 +3675,7 @@ static void builtin_diff(const char *name_a,
>> emit_diff_symbol(o, DIFF_SYMBOL_BINARY_FILES,
>> sb.buf, sb.len, 0);
>> strbuf_release(&sb);
>> + o->found_changes = 1;
>> goto free_ab_and_return;
>> }
>> if (fill_mmfile(o->repo, &mf1, one) < 0 ||
>
> I poked at the same issue in parallel and had the same fix, but ...
>
>> ...
>> test_expect_success "git diff $option returns 1 for copied file" "
>> git reset --hard &&
>> cp a copy &&
>
> your test is nicer.
>
> The patch works here locally.
>
> For what it's worth:
>
> Reviewed-by: Thomas Braun <thomas.braun@virtuell-zuhause.de>
>
> Thomas
Thanks.
^ permalink raw reply [flat|nested] 4+ messages in thread
end of thread, other threads:[~2024-09-25 21:52 UTC | newest]
Thread overview: 4+ messages (download: mbox.gz follow: Atom feed
-- links below jump to the message on this page --
2024-09-21 4:26 git diff --exit-code returns 0 when binary files differ Kohei Shibata
2024-09-21 15:09 ` [PATCH] diff: report modified binary files as changes in builtin_diff() René Scharfe
2024-09-25 21:24 ` Thomas Braun
2024-09-25 21:52 ` Junio C Hamano
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).