Merge branch 'mm/apply-reverse-mode-of-deleted-path'

"git apply --index/--cached" when applying a deletion patch in
reverse failed to give the mode bits of the path "removed" by the
patch to the file it creates, which has been corrected.

* mm/apply-reverse-mode-of-deleted-path:
  apply: set file mode when --reverse creates a deleted file
  t4129: test that git apply warns for unexpected mode changes
This commit is contained in:
Junio C Hamano
2025-05-30 11:59:17 -07:00
2 changed files with 227 additions and 6 deletions

View File

@@ -2219,7 +2219,7 @@ static void reverse_patches(struct patch *p)
struct fragment *frag = p->fragments; struct fragment *frag = p->fragments;
SWAP(p->new_name, p->old_name); SWAP(p->new_name, p->old_name);
if (p->new_mode) if (p->new_mode || p->is_delete)
SWAP(p->new_mode, p->old_mode); SWAP(p->new_mode, p->old_mode);
SWAP(p->is_new, p->is_delete); SWAP(p->is_new, p->is_delete);
SWAP(p->lines_added, p->lines_deleted); SWAP(p->lines_added, p->lines_deleted);

View File

@@ -102,15 +102,32 @@ test_expect_success POSIXPERM 'do not use core.sharedRepository for working tree
) )
' '
test_file_mode_common () {
if test "$1" = "000000"
then
test_must_be_empty "$2"
else
test_grep "^$1 " "$2"
fi
}
test_file_mode_staged () {
git ls-files --stage -- "$2" >ls-files-output &&
test_file_mode_common "$1" ls-files-output
}
test_file_mode_HEAD () {
git ls-tree HEAD -- "$2" >ls-tree-output &&
test_file_mode_common "$1" ls-tree-output
}
test_expect_success 'git apply respects core.fileMode' ' test_expect_success 'git apply respects core.fileMode' '
test_config core.fileMode false && test_config core.fileMode false &&
echo true >script.sh && echo true >script.sh &&
git add --chmod=+x script.sh && git add --chmod=+x script.sh &&
git ls-files -s script.sh >ls-files-output && test_file_mode_staged 100755 script.sh &&
test_grep "^100755" ls-files-output &&
test_tick && git commit -m "Add script" && test_tick && git commit -m "Add script" &&
git ls-tree -r HEAD script.sh >ls-tree-output && test_file_mode_HEAD 100755 script.sh &&
test_grep "^100755" ls-tree-output &&
echo true >>script.sh && echo true >>script.sh &&
test_tick && git commit -m "Modify script" script.sh && test_tick && git commit -m "Modify script" script.sh &&
@@ -126,7 +143,211 @@ test_expect_success 'git apply respects core.fileMode' '
test_grep ! "has type 100644, expected 100755" err && test_grep ! "has type 100644, expected 100755" err &&
git apply --cached patch 2>err && git apply --cached patch 2>err &&
test_grep ! "has type 100644, expected 100755" err test_grep ! "has type 100644, expected 100755" err &&
git reset --hard
'
test_expect_success 'setup: git apply [--reverse] warns about incorrect file modes' '
test_config core.fileMode false &&
>mode_test &&
git add --chmod=-x mode_test &&
test_file_mode_staged 100644 mode_test &&
test_tick && git commit -m "add mode_test" &&
test_file_mode_HEAD 100644 mode_test &&
git tag mode_test_forward_initial &&
echo content >>mode_test &&
test_tick && git commit -m "append to mode_test" mode_test &&
test_file_mode_HEAD 100644 mode_test &&
git tag mode_test_reverse_initial &&
git format-patch -1 --stdout >patch &&
test_grep "^index .* 100644$" patch
'
test_expect_success 'git apply warns about incorrect file modes' '
test_config core.fileMode false &&
git reset --hard mode_test_forward_initial &&
git add --chmod=+x mode_test &&
test_file_mode_staged 100755 mode_test &&
test_tick && git commit -m "make mode_test executable" &&
test_file_mode_HEAD 100755 mode_test &&
git apply --index patch 2>err &&
test_grep "has type 100755, expected 100644" err &&
test_file_mode_staged 100755 mode_test &&
test_tick && git commit -m "redo: append to mode_test" &&
test_file_mode_HEAD 100755 mode_test
'
test_expect_success 'git apply --reverse warns about incorrect file modes' '
test_config core.fileMode false &&
git reset --hard mode_test_reverse_initial &&
git add --chmod=+x mode_test &&
test_file_mode_staged 100755 mode_test &&
test_tick && git commit -m "make mode_test executable" &&
test_file_mode_HEAD 100755 mode_test &&
git apply --index --reverse patch 2>err &&
test_grep "has type 100755, expected 100644" err &&
test_file_mode_staged 100755 mode_test &&
test_tick && git commit -m "undo: append to mode_test" &&
test_file_mode_HEAD 100755 mode_test
'
test_expect_success 'setup: git apply [--reverse] restores file modes (change_x_to_notx)' '
test_config core.fileMode false &&
touch change_x_to_notx &&
git add --chmod=+x change_x_to_notx &&
test_file_mode_staged 100755 change_x_to_notx &&
test_tick && git commit -m "add change_x_to_notx as executable" &&
test_file_mode_HEAD 100755 change_x_to_notx &&
git add --chmod=-x change_x_to_notx &&
test_file_mode_staged 100644 change_x_to_notx &&
test_tick && git commit -m "make change_x_to_notx not executable" &&
test_file_mode_HEAD 100644 change_x_to_notx &&
git rm change_x_to_notx &&
test_file_mode_staged 000000 change_x_to_notx &&
test_tick && git commit -m "remove change_x_to_notx" &&
test_file_mode_HEAD 000000 change_x_to_notx &&
git format-patch -o patches -3 &&
mv patches/0001-* change_x_to_notx-0001-create-0755.patch &&
mv patches/0002-* change_x_to_notx-0002-chmod-0644.patch &&
mv patches/0003-* change_x_to_notx-0003-delete.patch &&
test_grep "^new file mode 100755$" change_x_to_notx-0001-create-0755.patch &&
test_grep "^old mode 100755$" change_x_to_notx-0002-chmod-0644.patch &&
test_grep "^new mode 100644$" change_x_to_notx-0002-chmod-0644.patch &&
test_grep "^deleted file mode 100644$" change_x_to_notx-0003-delete.patch &&
git tag change_x_to_notx_initial
'
test_expect_success 'git apply restores file modes (change_x_to_notx)' '
test_config core.fileMode false &&
git reset --hard change_x_to_notx_initial &&
git apply --index change_x_to_notx-0001-create-0755.patch &&
test_file_mode_staged 100755 change_x_to_notx &&
test_tick && git commit -m "redo: add change_x_to_notx as executable" &&
test_file_mode_HEAD 100755 change_x_to_notx &&
git apply --index change_x_to_notx-0002-chmod-0644.patch 2>err &&
test_grep ! "has type 100.*, expected 100.*" err &&
test_file_mode_staged 100644 change_x_to_notx &&
test_tick && git commit -m "redo: make change_x_to_notx not executable" &&
test_file_mode_HEAD 100644 change_x_to_notx &&
git apply --index change_x_to_notx-0003-delete.patch 2>err &&
test_grep ! "has type 100.*, expected 100.*" err &&
test_file_mode_staged 000000 change_x_to_notx &&
test_tick && git commit -m "redo: remove change_notx_to_x" &&
test_file_mode_HEAD 000000 change_x_to_notx
'
test_expect_success 'git apply --reverse restores file modes (change_x_to_notx)' '
test_config core.fileMode false &&
git reset --hard change_x_to_notx_initial &&
git apply --index --reverse change_x_to_notx-0003-delete.patch &&
test_file_mode_staged 100644 change_x_to_notx &&
test_tick && git commit -m "undo: remove change_x_to_notx" &&
test_file_mode_HEAD 100644 change_x_to_notx &&
git apply --index --reverse change_x_to_notx-0002-chmod-0644.patch 2>err &&
test_grep ! "has type 100.*, expected 100.*" err &&
test_file_mode_staged 100755 change_x_to_notx &&
test_tick && git commit -m "undo: make change_x_to_notx not executable" &&
test_file_mode_HEAD 100755 change_x_to_notx &&
git apply --index --reverse change_x_to_notx-0001-create-0755.patch 2>err &&
test_grep ! "has type 100.*, expected 100.*" err &&
test_file_mode_staged 000000 change_x_to_notx &&
test_tick && git commit -m "undo: add change_x_to_notx as executable" &&
test_file_mode_HEAD 000000 change_x_to_notx
'
test_expect_success 'setup: git apply [--reverse] restores file modes (change_notx_to_x)' '
test_config core.fileMode false &&
touch change_notx_to_x &&
git add --chmod=-x change_notx_to_x &&
test_file_mode_staged 100644 change_notx_to_x &&
test_tick && git commit -m "add change_notx_to_x as not executable" &&
test_file_mode_HEAD 100644 change_notx_to_x &&
git add --chmod=+x change_notx_to_x &&
test_file_mode_staged 100755 change_notx_to_x &&
test_tick && git commit -m "make change_notx_to_x executable" &&
test_file_mode_HEAD 100755 change_notx_to_x &&
git rm change_notx_to_x &&
test_file_mode_staged 000000 change_notx_to_x &&
test_tick && git commit -m "remove change_notx_to_x" &&
test_file_mode_HEAD 000000 change_notx_to_x &&
git format-patch -o patches -3 &&
mv patches/0001-* change_notx_to_x-0001-create-0644.patch &&
mv patches/0002-* change_notx_to_x-0002-chmod-0755.patch &&
mv patches/0003-* change_notx_to_x-0003-delete.patch &&
test_grep "^new file mode 100644$" change_notx_to_x-0001-create-0644.patch &&
test_grep "^old mode 100644$" change_notx_to_x-0002-chmod-0755.patch &&
test_grep "^new mode 100755$" change_notx_to_x-0002-chmod-0755.patch &&
test_grep "^deleted file mode 100755$" change_notx_to_x-0003-delete.patch &&
git tag change_notx_to_x_initial
'
test_expect_success 'git apply restores file modes (change_notx_to_x)' '
test_config core.fileMode false &&
git reset --hard change_notx_to_x_initial &&
git apply --index change_notx_to_x-0001-create-0644.patch &&
test_file_mode_staged 100644 change_notx_to_x &&
test_tick && git commit -m "redo: add change_notx_to_x as not executable" &&
test_file_mode_HEAD 100644 change_notx_to_x &&
git apply --index change_notx_to_x-0002-chmod-0755.patch 2>err &&
test_grep ! "has type 100.*, expected 100.*" err &&
test_file_mode_staged 100755 change_notx_to_x &&
test_tick && git commit -m "redo: make change_notx_to_x executable" &&
test_file_mode_HEAD 100755 change_notx_to_x &&
git apply --index change_notx_to_x-0003-delete.patch &&
test_grep ! "has type 100.*, expected 100.*" err &&
test_file_mode_staged 000000 change_notx_to_x &&
test_tick && git commit -m "undo: remove change_notx_to_x" &&
test_file_mode_HEAD 000000 change_notx_to_x
'
test_expect_success 'git apply --reverse restores file modes (change_notx_to_x)' '
test_config core.fileMode false &&
git reset --hard change_notx_to_x_initial &&
git apply --index --reverse change_notx_to_x-0003-delete.patch &&
test_file_mode_staged 100755 change_notx_to_x &&
test_tick && git commit -m "undo: remove change_notx_to_x" &&
test_file_mode_HEAD 100755 change_notx_to_x &&
git apply --index --reverse change_notx_to_x-0002-chmod-0755.patch 2>err &&
test_grep ! "has type 100.*, expected 100.*" err &&
test_file_mode_staged 100644 change_notx_to_x &&
test_tick && git commit -m "undo: make change_notx_to_x executable" &&
test_file_mode_HEAD 100644 change_notx_to_x &&
git apply --index --reverse change_notx_to_x-0001-create-0644.patch 2>err &&
test_grep ! "has type 100.*, expected 100.*" err &&
test_file_mode_staged 000000 change_notx_to_x &&
test_tick && git commit -m "undo: add change_notx_to_x as not executable" &&
test_file_mode_HEAD 000000 change_notx_to_x
' '
test_expect_success POSIXPERM 'patch mode for new file is canonicalized' ' test_expect_success POSIXPERM 'patch mode for new file is canonicalized' '