mirror of
https://github.com/git/git.git
synced 2025-01-25 08:54:08 +08:00
20b813d7d3
Since [1] first released with Git v2.37.0 the built-in version of "add -i" has been the default. That built-in implementation was added in [2], first released with Git v2.25.0. At this point enough time has passed to allow for finding any remaining bugs in this new implementation, so let's remove the fallback code. As with similar migrations for "stash"[3] and "rebase"[4] we're keeping a mention of "add.interactive.useBuiltin" in the documentation, but adding a warning() to notify any outstanding users that the built-in is now the default. As with [5] and [6] we should follow-up in the future and eventually remove that warning. 1.0527ccb1b5
(add -i: default to the built-in implementation, 2021-11-30) 2.f83dff60a7
(Start to implement a built-in version of `git add --interactive`, 2019-11-13) 3.8a2cd3f512
(stash: remove the stash.useBuiltin setting, 2020-03-03) 4.d03ebd411c
(rebase: remove the rebase.useBuiltin setting, 2019-03-18) 5.deeaf5ee07
(stash: remove documentation for `stash.useBuiltin`, 2022-01-27) 6.9bcde4d531
(rebase: remove transitory rebase.useBuiltin setting & env, 2021-03-23) Signed-off-by: Ævar Arnfjörð Bjarmason <avarab@gmail.com> Signed-off-by: Junio C Hamano <gitster@pobox.com>
429 lines
9.2 KiB
Bash
Executable File
429 lines
9.2 KiB
Bash
Executable File
#!/bin/sh
|
|
|
|
test_description='test case exclude pathspec'
|
|
|
|
. ./test-lib.sh
|
|
|
|
test_expect_success 'setup' '
|
|
for p in file sub/file sub/sub/file sub/file2 sub/sub/sub/file sub2/file; do
|
|
if echo $p | grep /; then
|
|
mkdir -p $(dirname $p)
|
|
fi &&
|
|
: >$p &&
|
|
git add $p &&
|
|
git commit -m $p || return 1
|
|
done &&
|
|
git log --oneline --format=%s >actual &&
|
|
cat <<EOF >expect &&
|
|
sub2/file
|
|
sub/sub/sub/file
|
|
sub/file2
|
|
sub/sub/file
|
|
sub/file
|
|
file
|
|
EOF
|
|
test_cmp expect actual
|
|
'
|
|
|
|
test_expect_success 'exclude only pathspec uses default implicit pathspec' '
|
|
git log --oneline --format=%s -- . ":(exclude)sub" >expect &&
|
|
git log --oneline --format=%s -- ":(exclude)sub" >actual &&
|
|
test_cmp expect actual
|
|
'
|
|
|
|
test_expect_success 't_e_i() exclude sub' '
|
|
git log --oneline --format=%s -- . ":(exclude)sub" >actual &&
|
|
cat <<EOF >expect &&
|
|
sub2/file
|
|
file
|
|
EOF
|
|
test_cmp expect actual
|
|
'
|
|
|
|
test_expect_success 't_e_i() exclude sub/sub/file' '
|
|
git log --oneline --format=%s -- . ":(exclude)sub/sub/file" >actual &&
|
|
cat <<EOF >expect &&
|
|
sub2/file
|
|
sub/sub/sub/file
|
|
sub/file2
|
|
sub/file
|
|
file
|
|
EOF
|
|
test_cmp expect actual
|
|
'
|
|
|
|
test_expect_success 't_e_i() exclude sub using mnemonic' '
|
|
git log --oneline --format=%s -- . ":!sub" >actual &&
|
|
cat <<EOF >expect &&
|
|
sub2/file
|
|
file
|
|
EOF
|
|
test_cmp expect actual
|
|
'
|
|
|
|
test_expect_success 't_e_i() exclude :(icase)SUB' '
|
|
git log --oneline --format=%s -- . ":(exclude,icase)SUB" >actual &&
|
|
cat <<EOF >expect &&
|
|
sub2/file
|
|
file
|
|
EOF
|
|
test_cmp expect actual
|
|
'
|
|
|
|
test_expect_success 't_e_i() exclude sub2 from sub' '
|
|
(
|
|
cd sub &&
|
|
git log --oneline --format=%s -- :/ ":/!sub2" >actual &&
|
|
cat <<EOF >expect &&
|
|
sub/sub/sub/file
|
|
sub/file2
|
|
sub/sub/file
|
|
sub/file
|
|
file
|
|
EOF
|
|
test_cmp expect actual
|
|
)
|
|
'
|
|
|
|
test_expect_success 't_e_i() exclude sub/*file' '
|
|
git log --oneline --format=%s -- . ":(exclude)sub/*file" >actual &&
|
|
cat <<EOF >expect &&
|
|
sub2/file
|
|
sub/file2
|
|
file
|
|
EOF
|
|
test_cmp expect actual
|
|
'
|
|
|
|
test_expect_success 't_e_i() exclude :(glob)sub/*/file' '
|
|
git log --oneline --format=%s -- . ":(exclude,glob)sub/*/file" >actual &&
|
|
cat <<EOF >expect &&
|
|
sub2/file
|
|
sub/sub/sub/file
|
|
sub/file2
|
|
sub/file
|
|
file
|
|
EOF
|
|
test_cmp expect actual
|
|
'
|
|
|
|
test_expect_success 'm_p_d() exclude sub' '
|
|
git ls-files -- . ":(exclude)sub" >actual &&
|
|
cat <<EOF >expect &&
|
|
file
|
|
sub2/file
|
|
EOF
|
|
test_cmp expect actual
|
|
'
|
|
|
|
test_expect_success 'm_p_d() exclude sub/sub/file' '
|
|
git ls-files -- . ":(exclude)sub/sub/file" >actual &&
|
|
cat <<EOF >expect &&
|
|
file
|
|
sub/file
|
|
sub/file2
|
|
sub/sub/sub/file
|
|
sub2/file
|
|
EOF
|
|
test_cmp expect actual
|
|
'
|
|
|
|
test_expect_success 'm_p_d() exclude sub using mnemonic' '
|
|
git ls-files -- . ":!sub" >actual &&
|
|
cat <<EOF >expect &&
|
|
file
|
|
sub2/file
|
|
EOF
|
|
test_cmp expect actual
|
|
'
|
|
|
|
test_expect_success 'm_p_d() exclude :(icase)SUB' '
|
|
git ls-files -- . ":(exclude,icase)SUB" >actual &&
|
|
cat <<EOF >expect &&
|
|
file
|
|
sub2/file
|
|
EOF
|
|
test_cmp expect actual
|
|
'
|
|
|
|
test_expect_success 'm_p_d() exclude sub2 from sub' '
|
|
(
|
|
cd sub &&
|
|
git ls-files -- :/ ":/!sub2" >actual &&
|
|
cat <<EOF >expect &&
|
|
../file
|
|
file
|
|
file2
|
|
sub/file
|
|
sub/sub/file
|
|
EOF
|
|
test_cmp expect actual
|
|
)
|
|
'
|
|
|
|
test_expect_success 'm_p_d() exclude sub/*file' '
|
|
git ls-files -- . ":(exclude)sub/*file" >actual &&
|
|
cat <<EOF >expect &&
|
|
file
|
|
sub/file2
|
|
sub2/file
|
|
EOF
|
|
test_cmp expect actual
|
|
'
|
|
|
|
test_expect_success 'm_p_d() exclude :(glob)sub/*/file' '
|
|
git ls-files -- . ":(exclude,glob)sub/*/file" >actual &&
|
|
cat <<EOF >expect &&
|
|
file
|
|
sub/file
|
|
sub/file2
|
|
sub/sub/sub/file
|
|
sub2/file
|
|
EOF
|
|
test_cmp expect actual
|
|
'
|
|
|
|
test_expect_success 'multiple exclusions' '
|
|
git ls-files -- ":^*/file2" ":^sub2" >actual &&
|
|
cat <<-\EOF >expect &&
|
|
file
|
|
sub/file
|
|
sub/sub/file
|
|
sub/sub/sub/file
|
|
EOF
|
|
test_cmp expect actual
|
|
'
|
|
|
|
test_expect_success 't_e_i() exclude case #8' '
|
|
test_when_finished "rm -fr case8" &&
|
|
git init case8 &&
|
|
(
|
|
cd case8 &&
|
|
echo file >file1 &&
|
|
echo file >file2 &&
|
|
git add file1 file2 &&
|
|
git commit -m twofiles &&
|
|
git grep -l file HEAD :^file2 >actual &&
|
|
echo HEAD:file1 >expected &&
|
|
test_cmp expected actual &&
|
|
git grep -l file HEAD :^file1 >actual &&
|
|
echo HEAD:file2 >expected &&
|
|
test_cmp expected actual
|
|
)
|
|
'
|
|
|
|
test_expect_success 'grep --untracked PATTERN' '
|
|
# This test is not an actual test of exclude patterns, rather it
|
|
# is here solely to ensure that if any tests are inserted, deleted, or
|
|
# changed above, that we still have untracked files with the expected
|
|
# contents for the NEXT two tests.
|
|
cat <<-\EOF >expect-grep &&
|
|
actual
|
|
expect
|
|
sub/actual
|
|
sub/expect
|
|
EOF
|
|
git grep -l --untracked file -- >actual-grep &&
|
|
test_cmp expect-grep actual-grep
|
|
'
|
|
|
|
test_expect_success 'grep --untracked PATTERN :(exclude)DIR' '
|
|
cat <<-\EOF >expect-grep &&
|
|
actual
|
|
expect
|
|
EOF
|
|
git grep -l --untracked file -- ":(exclude)sub" >actual-grep &&
|
|
test_cmp expect-grep actual-grep
|
|
'
|
|
|
|
test_expect_success 'grep --untracked PATTERN :(exclude)*FILE' '
|
|
cat <<-\EOF >expect-grep &&
|
|
actual
|
|
sub/actual
|
|
EOF
|
|
git grep -l --untracked file -- ":(exclude)*expect" >actual-grep &&
|
|
test_cmp expect-grep actual-grep
|
|
'
|
|
|
|
# Depending on the command, all negative pathspec needs to subtract
|
|
# either from the full tree, or from the current directory.
|
|
#
|
|
# The sample tree checked out at this point has:
|
|
# file
|
|
# sub/file
|
|
# sub/file2
|
|
# sub/sub/file
|
|
# sub/sub/sub/file
|
|
# sub2/file
|
|
#
|
|
# but there may also be some cruft that interferes with "git clean"
|
|
# and "git add" tests.
|
|
|
|
test_expect_success 'archive with all negative' '
|
|
git reset --hard &&
|
|
git clean -f &&
|
|
git -C sub archive --format=tar HEAD -- ":!sub/" >archive &&
|
|
"$TAR" tf archive >actual &&
|
|
cat >expect <<-\EOF &&
|
|
file
|
|
file2
|
|
EOF
|
|
test_cmp expect actual
|
|
'
|
|
|
|
test_expect_success 'add with all negative' '
|
|
H=$(git rev-parse HEAD) &&
|
|
git reset --hard $H &&
|
|
git clean -f &&
|
|
test_when_finished "git reset --hard $H" &&
|
|
for path in file sub/file sub/sub/file sub2/file
|
|
do
|
|
echo smudge >>"$path" || return 1
|
|
done &&
|
|
git -C sub add -- ":!sub/" &&
|
|
git diff --name-only --no-renames --cached >actual &&
|
|
cat >expect <<-\EOF &&
|
|
file
|
|
sub/file
|
|
sub2/file
|
|
EOF
|
|
test_cmp expect actual &&
|
|
git diff --name-only --no-renames >actual &&
|
|
echo sub/sub/file >expect &&
|
|
test_cmp expect actual
|
|
'
|
|
|
|
test_expect_success 'add -p with all negative' '
|
|
H=$(git rev-parse HEAD) &&
|
|
git reset --hard $H &&
|
|
git clean -f &&
|
|
test_when_finished "git reset --hard $H" &&
|
|
for path in file sub/file sub/sub/file sub2/file
|
|
do
|
|
echo smudge >>"$path" || return 1
|
|
done &&
|
|
yes | git -C sub add -p -- ":!sub/" &&
|
|
git diff --name-only --no-renames --cached >actual &&
|
|
cat >expect <<-\EOF &&
|
|
file
|
|
sub/file
|
|
sub2/file
|
|
EOF
|
|
test_cmp expect actual &&
|
|
git diff --name-only --no-renames >actual &&
|
|
echo sub/sub/file >expect &&
|
|
test_cmp expect actual
|
|
'
|
|
|
|
test_expect_success 'clean with all negative' '
|
|
H=$(git rev-parse HEAD) &&
|
|
git reset --hard $H &&
|
|
test_when_finished "git reset --hard $H && git clean -f" &&
|
|
git clean -f &&
|
|
for path in file9 sub/file9 sub/sub/file9 sub2/file9
|
|
do
|
|
echo cruft >"$path" || return 1
|
|
done &&
|
|
git -C sub clean -f -- ":!sub" &&
|
|
test_path_is_file file9 &&
|
|
test_path_is_missing sub/file9 &&
|
|
test_path_is_file sub/sub/file9 &&
|
|
test_path_is_file sub2/file9
|
|
'
|
|
|
|
test_expect_success 'commit with all negative' '
|
|
H=$(git rev-parse HEAD) &&
|
|
git reset --hard $H &&
|
|
test_when_finished "git reset --hard $H" &&
|
|
for path in file sub/file sub/sub/file sub2/file
|
|
do
|
|
echo smudge >>"$path" || return 1
|
|
done &&
|
|
git -C sub commit -m sample -- ":!sub/" &&
|
|
git diff --name-only --no-renames HEAD^ HEAD >actual &&
|
|
cat >expect <<-\EOF &&
|
|
file
|
|
sub/file
|
|
sub2/file
|
|
EOF
|
|
test_cmp expect actual &&
|
|
git diff --name-only --no-renames HEAD >actual &&
|
|
echo sub/sub/file >expect &&
|
|
test_cmp expect actual
|
|
'
|
|
|
|
test_expect_success 'reset with all negative' '
|
|
H=$(git rev-parse HEAD) &&
|
|
git reset --hard $H &&
|
|
test_when_finished "git reset --hard $H" &&
|
|
for path in file sub/file sub/sub/file sub2/file
|
|
do
|
|
echo smudge >>"$path" &&
|
|
git add "$path" || return 1
|
|
done &&
|
|
git -C sub reset --quiet -- ":!sub/" &&
|
|
git diff --name-only --no-renames --cached >actual &&
|
|
echo sub/sub/file >expect &&
|
|
test_cmp expect actual
|
|
'
|
|
|
|
test_expect_success 'grep with all negative' '
|
|
H=$(git rev-parse HEAD) &&
|
|
git reset --hard $H &&
|
|
test_when_finished "git reset --hard $H" &&
|
|
for path in file sub/file sub/sub/file sub2/file
|
|
do
|
|
echo "needle $path" >>"$path" || return 1
|
|
done &&
|
|
git -C sub grep -h needle -- ":!sub/" >actual &&
|
|
cat >expect <<-\EOF &&
|
|
needle sub/file
|
|
EOF
|
|
test_cmp expect actual
|
|
'
|
|
|
|
test_expect_success 'ls-files with all negative' '
|
|
git reset --hard &&
|
|
git -C sub ls-files -- ":!sub/" >actual &&
|
|
cat >expect <<-\EOF &&
|
|
file
|
|
file2
|
|
EOF
|
|
test_cmp expect actual
|
|
'
|
|
|
|
test_expect_success 'rm with all negative' '
|
|
git reset --hard &&
|
|
test_when_finished "git reset --hard" &&
|
|
git -C sub rm -r --cached -- ":!sub/" >actual &&
|
|
git diff --name-only --no-renames --diff-filter=D --cached >actual &&
|
|
cat >expect <<-\EOF &&
|
|
sub/file
|
|
sub/file2
|
|
EOF
|
|
test_cmp expect actual
|
|
'
|
|
|
|
test_expect_success 'stash with all negative' '
|
|
H=$(git rev-parse HEAD) &&
|
|
git reset --hard $H &&
|
|
test_when_finished "git reset --hard $H" &&
|
|
for path in file sub/file sub/sub/file sub2/file
|
|
do
|
|
echo smudge >>"$path" || return 1
|
|
done &&
|
|
git -C sub stash push -m sample -- ":!sub/" &&
|
|
git diff --name-only --no-renames HEAD >actual &&
|
|
echo sub/sub/file >expect &&
|
|
test_cmp expect actual &&
|
|
git stash show --name-only >actual &&
|
|
cat >expect <<-\EOF &&
|
|
file
|
|
sub/file
|
|
sub2/file
|
|
EOF
|
|
test_cmp expect actual
|
|
'
|
|
|
|
test_done
|