#!/bin/sh test_description='test labels in pathspecs' TEST_PASSES_SANITIZE_LEAK=true . ./test-lib.sh test_expect_success 'setup a tree' ' cat <<-\EOF >expect && fileA fileAB fileAC fileB fileBC fileC fileNoLabel fileSetLabel fileUnsetLabel fileValue fileWrongLabel sub/fileA sub/fileAB sub/fileAC sub/fileB sub/fileBC sub/fileC sub/fileNoLabel sub/fileSetLabel sub/fileUnsetLabel sub/fileValue sub/fileWrongLabel EOF mkdir sub && while read path do echo content >$path && git add $path || return 1 done actual && test_cmp expect actual ' test_expect_success 'pathspec with no attr' ' test_must_fail git ls-files ":(attr:)" ' test_expect_success 'pathspec with labels and non existent .gitattributes' ' git ls-files ":(attr:label)" >actual && test_must_be_empty actual ' test_expect_success 'pathspec with labels and non existent .gitattributes (2)' ' test_must_fail git grep content HEAD -- ":(attr:label)" ' test_expect_success 'setup .gitattributes' ' cat <<-\EOF >.gitattributes && fileA labelA fileB labelB fileC labelC fileAB labelA labelB fileAC labelA labelC fileBC labelB labelC fileUnsetLabel -label fileSetLabel label fileValue label=foo fileWrongLabel label☺ newFileA* labelA newFileB* labelB EOF echo fileSetLabel label1 >sub/.gitattributes && git add .gitattributes sub/.gitattributes && git commit -m "add attributes" ' test_expect_success 'setup .gitignore' ' cat <<-\EOF >.gitignore && actual expect pathspec_file EOF git add .gitignore && git commit -m "add gitignore" ' test_expect_success 'check specific set attr' ' cat <<-\EOF >expect && fileSetLabel sub/fileSetLabel EOF git ls-files ":(attr:label)" >actual && test_cmp expect actual ' test_expect_success 'check set attr with pathspec pattern' ' echo sub/fileSetLabel >expect && git ls-files ":(attr:label)sub" >actual && test_cmp expect actual && git ls-files ":(attr:label)sub/" >actual && test_cmp expect actual ' test_expect_success 'check specific set attr in tree-ish' ' cat <<-\EOF >expect && HEAD:fileSetLabel HEAD:sub/fileSetLabel EOF git grep -l content HEAD ":(attr:label)" >actual && test_cmp expect actual ' test_expect_success 'check specific set attr with pathspec pattern in tree-ish' ' echo HEAD:sub/fileSetLabel >expect && git grep -l content HEAD ":(attr:label)sub" >actual && test_cmp expect actual && git grep -l content HEAD ":(attr:label)sub/" >actual && test_cmp expect actual ' test_expect_success 'check specific unset attr' ' cat <<-\EOF >expect && fileUnsetLabel sub/fileUnsetLabel EOF git ls-files ":(attr:-label)" >actual && test_cmp expect actual ' test_expect_success 'check specific unset attr (2)' ' cat <<-\EOF >expect && HEAD:fileUnsetLabel HEAD:sub/fileUnsetLabel EOF git grep -l content HEAD ":(attr:-label)" >actual && test_cmp expect actual ' test_expect_success 'check specific value attr' ' cat <<-\EOF >expect && fileValue sub/fileValue EOF git ls-files ":(attr:label=foo)" >actual && test_cmp expect actual && git ls-files ":(attr:label=bar)" >actual && test_must_be_empty actual ' test_expect_success 'check specific value attr (2)' ' cat <<-\EOF >expect && HEAD:fileValue HEAD:sub/fileValue EOF git grep -l content HEAD ":(attr:label=foo)" >actual && test_cmp expect actual && test_must_fail git grep -l content HEAD ":(attr:label=bar)" ' test_expect_success 'check unspecified attr' ' cat <<-\EOF >expect && .gitattributes .gitignore fileA fileAB fileAC fileB fileBC fileC fileNoLabel fileWrongLabel sub/.gitattributes sub/fileA sub/fileAB sub/fileAC sub/fileB sub/fileBC sub/fileC sub/fileNoLabel sub/fileWrongLabel EOF git ls-files ":(attr:!label)" >actual && test_cmp expect actual ' test_expect_success 'check unspecified attr (2)' ' cat <<-\EOF >expect && HEAD:.gitattributes HEAD:.gitignore HEAD:fileA HEAD:fileAB HEAD:fileAC HEAD:fileB HEAD:fileBC HEAD:fileC HEAD:fileNoLabel HEAD:fileWrongLabel HEAD:sub/.gitattributes HEAD:sub/fileA HEAD:sub/fileAB HEAD:sub/fileAC HEAD:sub/fileB HEAD:sub/fileBC HEAD:sub/fileC HEAD:sub/fileNoLabel HEAD:sub/fileWrongLabel EOF git grep -l ^ HEAD ":(attr:!label)" >actual && test_cmp expect actual ' test_expect_success 'check multiple unspecified attr' ' cat <<-\EOF >expect && .gitattributes .gitignore fileC fileNoLabel fileWrongLabel sub/.gitattributes sub/fileC sub/fileNoLabel sub/fileWrongLabel EOF git ls-files ":(attr:!labelB !labelA !label)" >actual && test_cmp expect actual ' test_expect_success 'check label with more labels but excluded path' ' cat <<-\EOF >expect && fileAB fileB fileBC EOF git ls-files ":(attr:labelB)" ":(exclude)sub/" >actual && test_cmp expect actual ' test_expect_success 'check label excluding other labels' ' cat <<-\EOF >expect && fileAB fileB fileBC sub/fileAB sub/fileB EOF git ls-files ":(attr:labelB)" ":(exclude,attr:labelC)sub/" >actual && test_cmp expect actual ' test_expect_success 'fail on multiple attr specifiers in one pathspec item' ' test_must_fail git ls-files . ":(attr:labelB,attr:labelC)" 2>actual && test_grep "Only one" actual ' test_expect_success 'fail if attr magic is used in places not implemented' ' # The main purpose of this test is to check that we actually fail # when you attempt to use attr magic in commands that do not implement # attr magic. This test does not advocate check-ignore to stay that way. # When you teach the command to grok the pathspec, you need to find # another command to replace it for the test. test_must_fail git check-ignore ":(attr:labelB)" 2>actual && test_grep "magic not supported" actual ' test_expect_success 'check that attr magic works for git stash push' ' cat <<-\EOF >expect && A sub/newFileA-foo EOF >sub/newFileA-foo && >sub/newFileB-foo && git stash push --include-untracked -- ":(exclude,attr:labelB)" && git stash show --include-untracked --name-status >actual && test_cmp expect actual ' test_expect_success 'check that attr magic works for git add --all' ' cat <<-\EOF >expect && sub/newFileA-foo EOF >sub/newFileA-foo && >sub/newFileB-foo && git add --all ":(exclude,attr:labelB)" && git diff --name-only --cached >actual && git restore -W -S . && test_cmp expect actual ' test_expect_success 'check that attr magic works for git add -u' ' cat <<-\EOF >expect && sub/fileA EOF >sub/newFileA-foo && >sub/newFileB-foo && >sub/fileA && >sub/fileB && git add -u ":(exclude,attr:labelB)" && git diff --name-only --cached >actual && git restore -S -W . && rm sub/new* && test_cmp expect actual ' test_expect_success 'check that attr magic works for git add ' ' cat <<-\EOF >expect && fileA fileB sub/fileA EOF >fileA && >fileB && >sub/fileA && >sub/fileB && git add ":(exclude,attr:labelB)sub/*" && git diff --name-only --cached >actual && git restore -S -W . && test_cmp expect actual ' test_expect_success 'check that attr magic works for git -add .' ' cat <<-\EOF >expect && sub/fileA EOF >fileA && >fileB && >sub/fileA && >sub/fileB && cd sub && git add . ":(exclude,attr:labelB)" && cd .. && git diff --name-only --cached >actual && git restore -S -W . && test_cmp expect actual ' test_expect_success 'check that attr magic works for git add --pathspec-from-file' ' cat <<-\EOF >pathspec_file && :(exclude,attr:labelB) EOF cat <<-\EOF >expect && sub/newFileA-foo EOF >sub/newFileA-foo && >sub/newFileB-foo && git add --all --pathspec-from-file=pathspec_file && git diff --name-only --cached >actual && test_cmp expect actual ' test_expect_success 'abort on giving invalid label on the command line' ' test_must_fail git ls-files . ":(attr:☺)" ' test_expect_success 'abort on asking for wrong magic' ' test_must_fail git ls-files . ":(attr:-label=foo)" && test_must_fail git ls-files . ":(attr:!label=foo)" ' test_expect_success 'check attribute list' ' cat <<-EOF >>.gitattributes && * whitespace=indent,trail,space EOF git ls-files ":(attr:whitespace=indent\,trail\,space)" >actual && git ls-files >expect && test_cmp expect actual ' test_expect_success 'backslash cannot be the last character' ' test_must_fail git ls-files ":(attr:label=foo\\ labelA=bar)" 2>actual && test_grep "not allowed as last character in attr value" actual ' test_expect_success 'backslash cannot be used as a value' ' test_must_fail git ls-files ":(attr:label=f\\\oo)" 2>actual && test_grep "for value matching" actual ' test_expect_success 'reading from .gitattributes in a subdirectory (1)' ' git ls-files ":(attr:label1)" >actual && test_write_lines "sub/fileSetLabel" >expect && test_cmp expect actual ' test_expect_success 'reading from .gitattributes in a subdirectory (2)' ' git ls-files ":(attr:label1)sub" >actual && test_write_lines "sub/fileSetLabel" >expect && test_cmp expect actual ' test_expect_success 'reading from .gitattributes in a subdirectory (3)' ' git ls-files ":(attr:label1)sub/" >actual && test_write_lines "sub/fileSetLabel" >expect && test_cmp expect actual ' test_expect_success POSIXPERM 'pathspec with builtin_objectmode attr can be used' ' >mode_exec_file_1 && git status -s ":(attr:builtin_objectmode=100644)mode_exec_*" >actual && echo ?? mode_exec_file_1 >expect && test_cmp expect actual && git add mode_exec_file_1 && chmod +x mode_exec_file_1 && git status -s ":(attr:builtin_objectmode=100755)mode_exec_*" >actual && echo AM mode_exec_file_1 >expect && test_cmp expect actual ' test_expect_success POSIXPERM 'builtin_objectmode attr can be excluded' ' >mode_1_regular && >mode_1_exec && chmod +x mode_1_exec && git status -s ":(exclude,attr:builtin_objectmode=100644)" "mode_1_*" >actual && echo ?? mode_1_exec >expect && test_cmp expect actual && git status -s ":(exclude,attr:builtin_objectmode=100755)" "mode_1_*" >actual && echo ?? mode_1_regular >expect && test_cmp expect actual ' test_done