mirror of
https://github.com/git/git.git
synced 2025-12-12 20:36:24 +01:00
Given a set of attribute macros like:
[attr]a1 a2
[attr]a2 a3
...
[attr]a300000 -text
file a1
expanding the attributes for "file" requires expanding "a1" to "a2",
"a2" to "a3", and so on until hitting a non-macro expansion ("-text", in
this case). We implement this via recursion: fill_one() calls
macroexpand_one(), which then recurses back to fill_one(). As a result,
very deep macro chains like the one above can run out of stack space and
cause us to segfault.
The required stack space is fairly small; I needed on the order of
200,000 entries to get a segfault on Linux. So it's unlikely anybody
would hit this accidentally, leaving only malicious inputs. There you
can easily construct a repo which will segfault on clone (we look at
attributes during the checkout step, but you'd see the same trying to do
other operations, like diff in a bare repo). It's mostly harmless, since
anybody constructing such a repo is only preventing victims from cloning
their evil garbage, but it could be a nuisance for hosting sites.
One option to prevent this is to limit the depth of recursion we'll
allow. This is conceptually easy to implement, but it raises other
questions: what should the limit be, and do we need a configuration knob
for it?
The recursion here is simple enough that we can avoid those questions by
just converting it to iteration instead. Rather than iterate over the
states of a match_attr in fill_one(), we'll put them all in a queue, and
the expansion of each can add to the queue rather than recursing. Note
that this is a LIFO queue in order to keep the same depth-first order we
did with the recursive implementation. I've avoided using the word
"stack" in the code because the term is already heavily used to refer to
the stack of .gitattribute files that matches the tree structure of the
repository.
The test uses a limited stack size so we can trigger the problem with a
much smaller input than the one shown above. The value here (3000) is
enough to trigger the issue on my x86_64 Linux machine.
Reported-by: Ben Stav <benstav@miggo.io>
Signed-off-by: Jeff King <peff@peff.net>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
688 lines
20 KiB
Bash
Executable File
688 lines
20 KiB
Bash
Executable File
#!/bin/sh
|
|
|
|
test_description=gitattributes
|
|
|
|
TEST_CREATE_REPO_NO_TEMPLATE=1
|
|
. ./test-lib.sh
|
|
|
|
attr_check_basic () {
|
|
path="$1" expect="$2" git_opts="$3" &&
|
|
|
|
git $git_opts check-attr test -- "$path" >actual 2>err &&
|
|
echo "$path: test: $expect" >expect &&
|
|
test_cmp expect actual
|
|
}
|
|
|
|
attr_check () {
|
|
attr_check_basic "$@" &&
|
|
test_must_be_empty err
|
|
}
|
|
|
|
attr_check_object_mode_basic () {
|
|
path="$1" &&
|
|
expect="$2" &&
|
|
check_opts="$3" &&
|
|
git check-attr $check_opts builtin_objectmode -- "$path" >actual 2>err &&
|
|
echo "$path: builtin_objectmode: $expect" >expect &&
|
|
test_cmp expect actual
|
|
}
|
|
|
|
attr_check_object_mode () {
|
|
attr_check_object_mode_basic "$@" &&
|
|
test_must_be_empty err
|
|
}
|
|
|
|
attr_check_quote () {
|
|
path="$1" quoted_path="$2" expect="$3" &&
|
|
|
|
git check-attr test -- "$path" >actual &&
|
|
echo "\"$quoted_path\": test: $expect" >expect &&
|
|
test_cmp expect actual
|
|
}
|
|
|
|
attr_check_source () {
|
|
path="$1" expect="$2" source="$3" git_opts="$4" &&
|
|
|
|
echo "$path: test: $expect" >expect &&
|
|
|
|
git $git_opts check-attr --source $source test -- "$path" >actual 2>err &&
|
|
test_cmp expect actual &&
|
|
test_must_be_empty err &&
|
|
|
|
git $git_opts --attr-source="$source" check-attr test -- "$path" >actual 2>err &&
|
|
test_cmp expect actual &&
|
|
test_must_be_empty err
|
|
|
|
git $git_opts -c "attr.tree=$source" check-attr test -- "$path" >actual 2>err &&
|
|
test_cmp expect actual &&
|
|
test_must_be_empty err
|
|
|
|
GIT_ATTR_SOURCE="$source" git $git_opts check-attr test -- "$path" >actual 2>err &&
|
|
test_cmp expect actual &&
|
|
test_must_be_empty err
|
|
}
|
|
|
|
test_expect_success 'open-quoted pathname' '
|
|
echo "\"a test=a" >.gitattributes &&
|
|
attr_check a unspecified
|
|
'
|
|
|
|
test_expect_success 'setup' '
|
|
mkdir -p a/b/d a/c b &&
|
|
(
|
|
echo "[attr]notest !test" &&
|
|
echo "\" d \" test=d" &&
|
|
echo " e test=e" &&
|
|
echo " e\" test=e" &&
|
|
echo "f test=f" &&
|
|
echo "a/i test=a/i" &&
|
|
echo "onoff test -test" &&
|
|
echo "offon -test test" &&
|
|
echo "no notest" &&
|
|
echo "A/e/F test=A/e/F"
|
|
) >.gitattributes &&
|
|
(
|
|
echo "g test=a/g" &&
|
|
echo "b/g test=a/b/g"
|
|
) >a/.gitattributes &&
|
|
(
|
|
echo "h test=a/b/h" &&
|
|
echo "d/* test=a/b/d/*" &&
|
|
echo "d/yes notest"
|
|
) >a/b/.gitattributes &&
|
|
(
|
|
echo "global test=global"
|
|
) >"$HOME"/global-gitattributes &&
|
|
cat <<-EOF >expect-all
|
|
f: test: f
|
|
a/f: test: f
|
|
a/c/f: test: f
|
|
a/g: test: a/g
|
|
a/b/g: test: a/b/g
|
|
b/g: test: unspecified
|
|
a/b/h: test: a/b/h
|
|
a/b/d/g: test: a/b/d/*
|
|
onoff: test: unset
|
|
offon: test: set
|
|
no: notest: set
|
|
no: test: unspecified
|
|
a/b/d/no: notest: set
|
|
a/b/d/no: test: a/b/d/*
|
|
a/b/d/yes: notest: set
|
|
a/b/d/yes: test: unspecified
|
|
EOF
|
|
'
|
|
|
|
test_expect_success 'setup branches' '
|
|
mkdir -p foo/bar &&
|
|
test_commit --printf "add .gitattributes" foo/bar/.gitattributes \
|
|
"f test=f\na/i test=n\n" tag-1 &&
|
|
test_commit --printf "add .gitattributes" foo/bar/.gitattributes \
|
|
"g test=g\na/i test=m\n" tag-2 &&
|
|
rm foo/bar/.gitattributes
|
|
'
|
|
|
|
test_expect_success 'command line checks' '
|
|
test_must_fail git check-attr &&
|
|
test_must_fail git check-attr -- &&
|
|
test_must_fail git check-attr test &&
|
|
test_must_fail git check-attr test -- &&
|
|
test_must_fail git check-attr -- f &&
|
|
test_must_fail git check-attr --source &&
|
|
test_must_fail git check-attr --source not-a-valid-ref &&
|
|
echo "f" | test_must_fail git check-attr --stdin &&
|
|
echo "f" | test_must_fail git check-attr --stdin -- f &&
|
|
echo "f" | test_must_fail git check-attr --stdin test -- f &&
|
|
test_must_fail git check-attr "" -- f
|
|
'
|
|
|
|
test_expect_success 'attribute test' '
|
|
|
|
attr_check " d " d &&
|
|
attr_check e e &&
|
|
attr_check_quote e\" e\\\" e &&
|
|
|
|
attr_check f f &&
|
|
attr_check a/f f &&
|
|
attr_check a/c/f f &&
|
|
attr_check a/g a/g &&
|
|
attr_check a/b/g a/b/g &&
|
|
attr_check b/g unspecified &&
|
|
attr_check a/b/h a/b/h &&
|
|
attr_check a/b/d/g "a/b/d/*" &&
|
|
attr_check onoff unset &&
|
|
attr_check offon set &&
|
|
attr_check no unspecified &&
|
|
attr_check a/b/d/no "a/b/d/*" &&
|
|
attr_check a/b/d/yes unspecified
|
|
'
|
|
|
|
test_expect_success 'attribute matching is case sensitive when core.ignorecase=0' '
|
|
|
|
attr_check F unspecified "-c core.ignorecase=0" &&
|
|
attr_check a/F unspecified "-c core.ignorecase=0" &&
|
|
attr_check a/c/F unspecified "-c core.ignorecase=0" &&
|
|
attr_check a/G unspecified "-c core.ignorecase=0" &&
|
|
attr_check a/B/g a/g "-c core.ignorecase=0" &&
|
|
attr_check a/b/G unspecified "-c core.ignorecase=0" &&
|
|
attr_check a/b/H unspecified "-c core.ignorecase=0" &&
|
|
attr_check a/b/D/g a/g "-c core.ignorecase=0" &&
|
|
attr_check oNoFf unspecified "-c core.ignorecase=0" &&
|
|
attr_check oFfOn unspecified "-c core.ignorecase=0" &&
|
|
attr_check NO unspecified "-c core.ignorecase=0" &&
|
|
attr_check a/b/D/NO unspecified "-c core.ignorecase=0" &&
|
|
attr_check a/b/d/YES a/b/d/* "-c core.ignorecase=0" &&
|
|
attr_check a/E/f f "-c core.ignorecase=0"
|
|
|
|
'
|
|
|
|
test_expect_success 'attribute matching is case insensitive when core.ignorecase=1' '
|
|
|
|
attr_check F f "-c core.ignorecase=1" &&
|
|
attr_check a/F f "-c core.ignorecase=1" &&
|
|
attr_check a/c/F f "-c core.ignorecase=1" &&
|
|
attr_check a/G a/g "-c core.ignorecase=1" &&
|
|
attr_check a/B/g a/b/g "-c core.ignorecase=1" &&
|
|
attr_check a/b/G a/b/g "-c core.ignorecase=1" &&
|
|
attr_check a/b/H a/b/h "-c core.ignorecase=1" &&
|
|
attr_check a/b/D/g "a/b/d/*" "-c core.ignorecase=1" &&
|
|
attr_check oNoFf unset "-c core.ignorecase=1" &&
|
|
attr_check oFfOn set "-c core.ignorecase=1" &&
|
|
attr_check NO unspecified "-c core.ignorecase=1" &&
|
|
attr_check a/b/D/NO "a/b/d/*" "-c core.ignorecase=1" &&
|
|
attr_check a/b/d/YES unspecified "-c core.ignorecase=1" &&
|
|
attr_check a/E/f "A/e/F" "-c core.ignorecase=1"
|
|
|
|
'
|
|
|
|
test_expect_success CASE_INSENSITIVE_FS 'additional case insensitivity tests' '
|
|
attr_check a/B/D/g a/g "-c core.ignorecase=0" &&
|
|
attr_check A/B/D/NO unspecified "-c core.ignorecase=0" &&
|
|
attr_check A/b/h a/b/h "-c core.ignorecase=1" &&
|
|
attr_check a/B/D/g "a/b/d/*" "-c core.ignorecase=1" &&
|
|
attr_check A/B/D/NO "a/b/d/*" "-c core.ignorecase=1"
|
|
'
|
|
|
|
test_expect_success 'unnormalized paths' '
|
|
attr_check ./f f &&
|
|
attr_check ./a/g a/g &&
|
|
attr_check a/./g a/g &&
|
|
attr_check a/c/../b/g a/b/g
|
|
'
|
|
|
|
test_expect_success 'relative paths' '
|
|
(cd a && attr_check ../f f) &&
|
|
(cd a && attr_check f f) &&
|
|
(cd a && attr_check i a/i) &&
|
|
(cd a && attr_check g a/g) &&
|
|
(cd a && attr_check b/g a/b/g) &&
|
|
(cd b && attr_check ../a/f f) &&
|
|
(cd b && attr_check ../a/g a/g) &&
|
|
(cd b && attr_check ../a/b/g a/b/g)
|
|
'
|
|
|
|
test_expect_success 'prefixes are not confused with leading directories' '
|
|
attr_check a_plus/g unspecified &&
|
|
cat >expect <<-\EOF &&
|
|
a/g: test: a/g
|
|
a_plus/g: test: unspecified
|
|
EOF
|
|
git check-attr test a/g a_plus/g >actual &&
|
|
test_cmp expect actual
|
|
'
|
|
|
|
test_expect_success 'core.attributesfile' '
|
|
attr_check global unspecified &&
|
|
git config core.attributesfile "$HOME/global-gitattributes" &&
|
|
attr_check global global &&
|
|
git config core.attributesfile "~/global-gitattributes" &&
|
|
attr_check global global &&
|
|
echo "global test=precedence" >>.gitattributes &&
|
|
attr_check global precedence
|
|
'
|
|
|
|
test_expect_success 'attribute test: read paths from stdin' '
|
|
grep -v notest <expect-all >expect &&
|
|
sed -e "s/:.*//" <expect | git check-attr --stdin test >actual &&
|
|
test_cmp expect actual
|
|
'
|
|
|
|
test_expect_success 'setup --all option' '
|
|
grep -v unspecified <expect-all | sort >specified-all &&
|
|
sed -e "s/:.*//" <expect-all | uniq >stdin-all
|
|
'
|
|
|
|
test_expect_success 'attribute test: --all option' '
|
|
git check-attr --stdin --all <stdin-all >tmp &&
|
|
sort tmp >actual &&
|
|
test_cmp specified-all actual
|
|
'
|
|
|
|
test_expect_success 'attribute test: --cached option' '
|
|
git check-attr --cached --stdin --all <stdin-all >tmp &&
|
|
sort tmp >actual &&
|
|
test_must_be_empty actual &&
|
|
git add .gitattributes a/.gitattributes a/b/.gitattributes &&
|
|
git check-attr --cached --stdin --all <stdin-all >tmp &&
|
|
sort tmp >actual &&
|
|
test_cmp specified-all actual
|
|
'
|
|
|
|
test_expect_success 'root subdir attribute test' '
|
|
attr_check a/i a/i &&
|
|
attr_check subdir/a/i unspecified
|
|
'
|
|
|
|
test_expect_success 'negative patterns' '
|
|
echo "!f test=bar" >.gitattributes &&
|
|
git check-attr test -- '"'"'!f'"'"' 2>errors &&
|
|
test_grep "Negative patterns are ignored" errors
|
|
'
|
|
|
|
test_expect_success 'patterns starting with exclamation' '
|
|
echo "\!f test=foo" >.gitattributes &&
|
|
attr_check "!f" foo
|
|
'
|
|
|
|
test_expect_success '"**" test' '
|
|
echo "**/f foo=bar" >.gitattributes &&
|
|
cat <<\EOF >expect &&
|
|
f: foo: bar
|
|
a/f: foo: bar
|
|
a/b/f: foo: bar
|
|
a/b/c/f: foo: bar
|
|
EOF
|
|
git check-attr foo -- "f" >actual 2>err &&
|
|
git check-attr foo -- "a/f" >>actual 2>>err &&
|
|
git check-attr foo -- "a/b/f" >>actual 2>>err &&
|
|
git check-attr foo -- "a/b/c/f" >>actual 2>>err &&
|
|
test_cmp expect actual &&
|
|
test_must_be_empty err
|
|
'
|
|
|
|
test_expect_success '"**" with no slashes test' '
|
|
echo "a**f foo=bar" >.gitattributes &&
|
|
git check-attr foo -- "f" >actual &&
|
|
cat <<\EOF >expect &&
|
|
f: foo: unspecified
|
|
af: foo: bar
|
|
axf: foo: bar
|
|
a/f: foo: unspecified
|
|
a/b/f: foo: unspecified
|
|
a/b/c/f: foo: unspecified
|
|
EOF
|
|
git check-attr foo -- "f" >actual 2>err &&
|
|
git check-attr foo -- "af" >>actual 2>err &&
|
|
git check-attr foo -- "axf" >>actual 2>err &&
|
|
git check-attr foo -- "a/f" >>actual 2>>err &&
|
|
git check-attr foo -- "a/b/f" >>actual 2>>err &&
|
|
git check-attr foo -- "a/b/c/f" >>actual 2>>err &&
|
|
test_cmp expect actual &&
|
|
test_must_be_empty err
|
|
'
|
|
|
|
test_expect_success 'using --git-dir and --work-tree' '
|
|
mkdir unreal real &&
|
|
git init real &&
|
|
echo "file test=in-real" >real/.gitattributes &&
|
|
(
|
|
cd unreal &&
|
|
attr_check file in-real "--git-dir ../real/.git --work-tree ../real"
|
|
)
|
|
'
|
|
|
|
test_expect_success 'using --source' '
|
|
attr_check_source foo/bar/f f tag-1 &&
|
|
attr_check_source foo/bar/a/i n tag-1 &&
|
|
attr_check_source foo/bar/f unspecified tag-2 &&
|
|
attr_check_source foo/bar/a/i m tag-2 &&
|
|
attr_check_source foo/bar/g g tag-2 &&
|
|
attr_check_source foo/bar/g unspecified tag-1
|
|
'
|
|
|
|
test_expect_success 'setup bare' '
|
|
git clone --template= --bare . bare.git
|
|
'
|
|
|
|
test_expect_success 'bare repository: check that .gitattribute is ignored' '
|
|
(
|
|
cd bare.git &&
|
|
(
|
|
echo "f test=f" &&
|
|
echo "a/i test=a/i"
|
|
) >.gitattributes &&
|
|
attr_check f unspecified &&
|
|
attr_check a/f unspecified &&
|
|
attr_check a/c/f unspecified &&
|
|
attr_check a/i unspecified &&
|
|
attr_check subdir/a/i unspecified
|
|
)
|
|
'
|
|
|
|
bad_attr_source_err="fatal: bad --attr-source or GIT_ATTR_SOURCE"
|
|
|
|
test_expect_success '--attr-source is bad' '
|
|
test_when_finished rm -rf empty &&
|
|
git init empty &&
|
|
(
|
|
cd empty &&
|
|
echo "$bad_attr_source_err" >expect_err &&
|
|
test_must_fail git --attr-source=HEAD check-attr test -- f/path 2>err &&
|
|
test_cmp expect_err err
|
|
)
|
|
'
|
|
|
|
test_expect_success 'attr.tree when HEAD is unborn' '
|
|
test_when_finished rm -rf empty &&
|
|
git init empty &&
|
|
(
|
|
cd empty &&
|
|
echo "f/path: test: unspecified" >expect &&
|
|
git -c attr.tree=HEAD check-attr test -- f/path >actual 2>err &&
|
|
test_must_be_empty err &&
|
|
test_cmp expect actual
|
|
)
|
|
'
|
|
|
|
test_expect_success 'bad attr source defaults to reading .gitattributes file' '
|
|
test_when_finished rm -rf empty &&
|
|
git init empty &&
|
|
(
|
|
cd empty &&
|
|
echo "f/path test=val" >.gitattributes &&
|
|
echo "f/path: test: val" >expect &&
|
|
git -c attr.tree=HEAD check-attr test -- f/path >actual 2>err &&
|
|
test_must_be_empty err &&
|
|
test_cmp expect actual
|
|
)
|
|
'
|
|
|
|
test_expect_success 'bare repo no longer defaults to reading .gitattributes from HEAD' '
|
|
test_when_finished rm -rf test bare_with_gitattribute &&
|
|
git init test &&
|
|
test_commit -C test gitattributes .gitattributes "f/path test=val" &&
|
|
git clone --bare test bare_with_gitattribute &&
|
|
|
|
echo "f/path: test: unspecified" >expect &&
|
|
git -C bare_with_gitattribute check-attr test -- f/path >actual &&
|
|
test_cmp expect actual &&
|
|
|
|
echo "f/path: test: val" >expect &&
|
|
git -C bare_with_gitattribute -c attr.tree=HEAD \
|
|
check-attr test -- f/path >actual &&
|
|
test_cmp expect actual
|
|
'
|
|
|
|
test_expect_success 'precedence of --attr-source, GIT_ATTR_SOURCE, then attr.tree' '
|
|
test_when_finished rm -rf empty &&
|
|
git init empty &&
|
|
(
|
|
cd empty &&
|
|
git checkout -b attr-source &&
|
|
test_commit "val1" .gitattributes "f/path test=val1" &&
|
|
git checkout -b attr-tree &&
|
|
test_commit "val2" .gitattributes "f/path test=val2" &&
|
|
git checkout attr-source &&
|
|
echo "f/path: test: val1" >expect &&
|
|
GIT_ATTR_SOURCE=attr-source git -c attr.tree=attr-tree --attr-source=attr-source \
|
|
check-attr test -- f/path >actual &&
|
|
test_cmp expect actual &&
|
|
GIT_ATTR_SOURCE=attr-source git -c attr.tree=attr-tree \
|
|
check-attr test -- f/path >actual &&
|
|
test_cmp expect actual
|
|
)
|
|
'
|
|
|
|
test_expect_success 'diff without repository with attr source' '
|
|
mkdir -p "$TRASH_DIRECTORY/outside/nongit" &&
|
|
(
|
|
cd "$TRASH_DIRECTORY/outside/nongit" &&
|
|
GIT_CEILING_DIRECTORIES="$TRASH_DIRECTORY/outside" &&
|
|
export GIT_CEILING_DIRECTORIES &&
|
|
touch file &&
|
|
cat >expect <<-EOF &&
|
|
fatal: cannot use --attr-source or GIT_ATTR_SOURCE without repo
|
|
EOF
|
|
test_must_fail env GIT_ATTR_SOURCE=HEAD git grep --no-index foo file 2>err &&
|
|
test_cmp expect err
|
|
)
|
|
'
|
|
|
|
test_expect_success 'bare repository: with --source' '
|
|
(
|
|
cd bare.git &&
|
|
attr_check_source foo/bar/f f tag-1 &&
|
|
attr_check_source foo/bar/a/i n tag-1 &&
|
|
attr_check_source foo/bar/f unspecified tag-2 &&
|
|
attr_check_source foo/bar/a/i m tag-2 &&
|
|
attr_check_source foo/bar/g g tag-2 &&
|
|
attr_check_source foo/bar/g unspecified tag-1
|
|
)
|
|
'
|
|
|
|
test_expect_success 'bare repository: check that --cached honors index' '
|
|
(
|
|
cd bare.git &&
|
|
GIT_INDEX_FILE=../.git/index \
|
|
git check-attr --cached --stdin --all <../stdin-all |
|
|
sort >actual &&
|
|
test_cmp ../specified-all actual
|
|
)
|
|
'
|
|
|
|
test_expect_success 'bare repository: test info/attributes' '
|
|
(
|
|
cd bare.git &&
|
|
mkdir info &&
|
|
(
|
|
echo "f test=f" &&
|
|
echo "a/i test=a/i"
|
|
) >info/attributes &&
|
|
attr_check f f &&
|
|
attr_check a/f f &&
|
|
attr_check a/c/f f &&
|
|
attr_check a/i a/i &&
|
|
attr_check subdir/a/i unspecified
|
|
)
|
|
'
|
|
|
|
test_expect_success 'binary macro expanded by -a' '
|
|
echo "file binary" >.gitattributes &&
|
|
cat >expect <<-\EOF &&
|
|
file: binary: set
|
|
file: diff: unset
|
|
file: merge: unset
|
|
file: text: unset
|
|
EOF
|
|
git check-attr -a file >actual &&
|
|
test_cmp expect actual
|
|
'
|
|
|
|
test_expect_success 'query binary macro directly' '
|
|
echo "file binary" >.gitattributes &&
|
|
echo file: binary: set >expect &&
|
|
git check-attr binary file >actual &&
|
|
test_cmp expect actual
|
|
'
|
|
|
|
test_expect_success SYMLINKS 'set up symlink tests' '
|
|
echo "* test" >attr &&
|
|
rm -f .gitattributes
|
|
'
|
|
|
|
test_expect_success SYMLINKS 'symlinks respected in core.attributesFile' '
|
|
test_when_finished "rm symlink" &&
|
|
ln -s attr symlink &&
|
|
test_config core.attributesFile "$(pwd)/symlink" &&
|
|
attr_check file set
|
|
'
|
|
|
|
test_expect_success SYMLINKS 'symlinks respected in info/attributes' '
|
|
test_when_finished "rm .git/info/attributes" &&
|
|
mkdir .git/info &&
|
|
ln -s ../../attr .git/info/attributes &&
|
|
attr_check file set
|
|
'
|
|
|
|
test_expect_success SYMLINKS 'symlinks not respected in-tree' '
|
|
test_when_finished "rm -rf .gitattributes subdir" &&
|
|
ln -s attr .gitattributes &&
|
|
mkdir subdir &&
|
|
ln -s ../attr subdir/.gitattributes &&
|
|
attr_check_basic subdir/file unspecified &&
|
|
test_grep "unable to access.*gitattributes" err
|
|
'
|
|
|
|
test_expect_success 'large attributes line ignored in tree' '
|
|
test_when_finished "rm .gitattributes" &&
|
|
printf "path %02043d" 1 >.gitattributes &&
|
|
git check-attr --all path >actual 2>err &&
|
|
echo "warning: ignoring overly long attributes line 1" >expect &&
|
|
test_cmp expect err &&
|
|
test_must_be_empty actual
|
|
'
|
|
|
|
test_expect_success 'large attributes line ignores trailing content in tree' '
|
|
test_when_finished "rm .gitattributes" &&
|
|
# older versions of Git broke lines at 2048 bytes; the 2045 bytes
|
|
# of 0-padding here is accounting for the three bytes of "a 1", which
|
|
# would knock "trailing" to the "next" line, where it would be
|
|
# erroneously parsed.
|
|
printf "a %02045dtrailing attribute\n" 1 >.gitattributes &&
|
|
git check-attr --all trailing >actual 2>err &&
|
|
echo "warning: ignoring overly long attributes line 1" >expect &&
|
|
test_cmp expect err &&
|
|
test_must_be_empty actual
|
|
'
|
|
|
|
test_expect_success EXPENSIVE 'large attributes file ignored in tree' '
|
|
test_when_finished "rm .gitattributes" &&
|
|
dd if=/dev/zero of=.gitattributes bs=1048576 count=101 2>/dev/null &&
|
|
git check-attr --all path >/dev/null 2>err &&
|
|
echo "warning: ignoring overly large gitattributes file ${SQ}.gitattributes${SQ}" >expect &&
|
|
test_cmp expect err
|
|
'
|
|
|
|
test_expect_success 'large attributes line ignored in index' '
|
|
test_when_finished "git update-index --remove .gitattributes" &&
|
|
blob=$(printf "path %02043d" 1 | git hash-object -w --stdin) &&
|
|
git update-index --add --cacheinfo 100644,$blob,.gitattributes &&
|
|
git check-attr --cached --all path >actual 2>err &&
|
|
echo "warning: ignoring overly long attributes line 1" >expect &&
|
|
test_cmp expect err &&
|
|
test_must_be_empty actual
|
|
'
|
|
|
|
test_expect_success 'large attributes line ignores trailing content in index' '
|
|
test_when_finished "git update-index --remove .gitattributes" &&
|
|
blob=$(printf "a %02045dtrailing attribute\n" 1 | git hash-object -w --stdin) &&
|
|
git update-index --add --cacheinfo 100644,$blob,.gitattributes &&
|
|
git check-attr --cached --all trailing >actual 2>err &&
|
|
echo "warning: ignoring overly long attributes line 1" >expect &&
|
|
test_cmp expect err &&
|
|
test_must_be_empty actual
|
|
'
|
|
|
|
test_expect_success EXPENSIVE 'large attributes file ignored in index' '
|
|
test_when_finished "git update-index --remove .gitattributes" &&
|
|
blob=$(dd if=/dev/zero bs=1048576 count=101 2>/dev/null | git hash-object -w --stdin) &&
|
|
git update-index --add --cacheinfo 100644,$blob,.gitattributes &&
|
|
git check-attr --cached --all path >/dev/null 2>err &&
|
|
echo "warning: ignoring overly large gitattributes blob ${SQ}.gitattributes${SQ}" >expect &&
|
|
test_cmp expect err
|
|
'
|
|
|
|
test_expect_success EXPENSIVE 'large attributes blob ignored' '
|
|
test_when_finished "git update-index --remove .gitattributes" &&
|
|
blob=$(dd if=/dev/zero bs=1048576 count=101 2>/dev/null | git hash-object -w --stdin) &&
|
|
git update-index --add --cacheinfo 100644,$blob,.gitattributes &&
|
|
tree="$(git write-tree)" &&
|
|
git check-attr --cached --all --source="$tree" path >/dev/null 2>err &&
|
|
echo "warning: ignoring overly large gitattributes blob ${SQ}.gitattributes${SQ}" >expect &&
|
|
test_cmp expect err
|
|
'
|
|
|
|
test_expect_success 'builtin object mode attributes work (dir and regular paths)' '
|
|
>normal &&
|
|
attr_check_object_mode normal 100644 &&
|
|
mkdir dir &&
|
|
attr_check_object_mode dir 040000
|
|
'
|
|
|
|
test_expect_success POSIXPERM 'builtin object mode attributes work (executable)' '
|
|
>exec &&
|
|
chmod +x exec &&
|
|
attr_check_object_mode exec 100755
|
|
'
|
|
|
|
test_expect_success SYMLINKS 'builtin object mode attributes work (symlinks)' '
|
|
ln -s to_sym sym &&
|
|
attr_check_object_mode sym 120000
|
|
'
|
|
|
|
test_expect_success 'native object mode attributes work with --cached' '
|
|
>normal &&
|
|
git add normal &&
|
|
empty_blob=$(git rev-parse :normal) &&
|
|
git update-index --index-info <<-EOF &&
|
|
100755 $empty_blob 0 exec
|
|
120000 $empty_blob 0 symlink
|
|
EOF
|
|
attr_check_object_mode normal 100644 --cached &&
|
|
attr_check_object_mode exec 100755 --cached &&
|
|
attr_check_object_mode symlink 120000 --cached
|
|
'
|
|
|
|
test_expect_success 'check object mode attributes work for submodules' '
|
|
mkdir sub &&
|
|
(
|
|
cd sub &&
|
|
git init &&
|
|
mv .git .real &&
|
|
echo "gitdir: .real" >.git &&
|
|
test_commit first
|
|
) &&
|
|
attr_check_object_mode sub 160000 &&
|
|
attr_check_object_mode sub unspecified --cached &&
|
|
git add sub &&
|
|
attr_check_object_mode sub 160000 --cached
|
|
'
|
|
|
|
test_expect_success 'we do not allow user defined builtin_* attributes' '
|
|
echo "foo* builtin_foo" >.gitattributes &&
|
|
git add .gitattributes 2>actual &&
|
|
echo "builtin_foo is not a valid attribute name: .gitattributes:1" >expect &&
|
|
test_cmp expect actual
|
|
'
|
|
|
|
test_expect_success 'user defined builtin_objectmode values are ignored' '
|
|
echo "foo* builtin_objectmode=12345" >.gitattributes &&
|
|
git add .gitattributes &&
|
|
>foo_1 &&
|
|
attr_check_object_mode_basic foo_1 100644 &&
|
|
echo "builtin_objectmode is not a valid attribute name: .gitattributes:1" >expect &&
|
|
test_cmp expect err
|
|
'
|
|
|
|
test_expect_success ULIMIT_STACK_SIZE 'deep macro recursion' '
|
|
n=3000 &&
|
|
{
|
|
i=0 &&
|
|
while test $i -lt $n; do
|
|
echo "[attr]a$i a$((i+1))" &&
|
|
i=$((i+1)) ||
|
|
return 1
|
|
done &&
|
|
echo "[attr]a$n -text" &&
|
|
echo "file a0"
|
|
} >.gitattributes &&
|
|
{
|
|
echo "file: text: unset" &&
|
|
test_seq -f "file: a%d: set" 0 $n
|
|
} >expect &&
|
|
run_with_limited_stack git check-attr -a file >actual &&
|
|
test_cmp expect actual
|
|
'
|
|
|
|
test_done
|