convert: unify the "auto" handling of CRLF

Before this change,
$ echo "* text=auto" >.gitattributes
$ echo "* eol=crlf" >>.gitattributes

would have the same effect as
$ echo "* text" >.gitattributes
$ git config core.eol crlf

Since the 'eol' attribute had higher priority than 'text=auto', this may
corrupt binary files and is not what most users expect to happen.

Make the 'eol' attribute to obey 'text=auto' and now
$ echo "* text=auto" >.gitattributes
$ echo "* eol=crlf" >>.gitattributes
behaves the same as
$ echo "* text=auto" >.gitattributes
$ git config core.eol crlf

In other words,
$ echo "* text=auto eol=crlf" >.gitattributes
has the same effect as
$ git config core.autocrlf true

and
$ echo "* text=auto eol=lf" >.gitattributes
has the same effect as
$ git config core.autocrlf input

Signed-off-by: Torsten Bögershausen <tboegi@web.de>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
This commit is contained in:
Torsten Bögershausen 2016-06-28 10:01:13 +02:00 committed by Junio C Hamano
parent caa47adc5a
commit 6523728499
7 changed files with 69 additions and 59 deletions

View File

@ -389,13 +389,11 @@ file with mixed line endings would be reported by the `core.safecrlf`
mechanism. mechanism.
core.autocrlf:: core.autocrlf::
Setting this variable to "true" is almost the same as setting Setting this variable to "true" is the same as setting
the `text` attribute to "auto" on all files except that text the `text` attribute to "auto" on all files and core.eol to "crlf".
files are not guaranteed to be normalized: files that contain Set to true if you want to have `CRLF` line endings in your
`CRLF` in the repository will not be touched. Use this working directory and the repository has LF line endings.
setting if you want to have `CRLF` line endings in your This variable can be set to 'input',
working directory even though the repository does not have
normalized line endings. This variable can be set to 'input',
in which case no output conversion is performed. in which case no output conversion is performed.
core.symlinks:: core.symlinks::

View File

@ -115,6 +115,7 @@ text file is normalized, its line endings are converted to LF in the
repository. To control what line ending style is used in the working repository. To control what line ending style is used in the working
directory, use the `eol` attribute for a single file and the directory, use the `eol` attribute for a single file and the
`core.eol` configuration variable for all text files. `core.eol` configuration variable for all text files.
Note that `core.autocrlf` overrides `core.eol`
Set:: Set::
@ -130,8 +131,9 @@ Unset::
Set to string value "auto":: Set to string value "auto"::
When `text` is set to "auto", the path is marked for automatic When `text` is set to "auto", the path is marked for automatic
end-of-line normalization. If Git decides that the content is end-of-line conversion. If Git decides that the content is
text, its line endings are normalized to LF on checkin. text, its line endings are converted to LF on checkin.
When the file has been commited with CRLF, no conversion is done.
Unspecified:: Unspecified::
@ -146,7 +148,7 @@ unspecified.
^^^^^ ^^^^^
This attribute sets a specific line-ending style to be used in the This attribute sets a specific line-ending style to be used in the
working directory. It enables end-of-line normalization without any working directory. It enables end-of-line conversion without any
content checks, effectively setting the `text` attribute. content checks, effectively setting the `text` attribute.
Set to string value "crlf":: Set to string value "crlf"::
@ -186,9 +188,10 @@ the working directory, and prevent .jpg files from being normalized
regardless of their content. regardless of their content.
------------------------ ------------------------
* text=auto
*.txt text *.txt text
*.vcproj eol=crlf *.vcproj text eol=crlf
*.sh eol=lf *.sh text eol=lf
*.jpg -text *.jpg -text
------------------------ ------------------------
@ -198,7 +201,7 @@ normalization in Git.
If you simply want to have CRLF line endings in your working directory If you simply want to have CRLF line endings in your working directory
regardless of the repository you are working with, you can set the regardless of the repository you are working with, you can set the
config variable "core.autocrlf" without changing any attributes. config variable "core.autocrlf" without using any attributes.
------------------------ ------------------------
[core] [core]

View File

@ -176,7 +176,9 @@ static enum eol output_eol(enum crlf_action crlf_action)
return EOL_LF; return EOL_LF;
case CRLF_UNDEFINED: case CRLF_UNDEFINED:
case CRLF_AUTO_CRLF: case CRLF_AUTO_CRLF:
return EOL_CRLF;
case CRLF_AUTO_INPUT: case CRLF_AUTO_INPUT:
return EOL_LF;
case CRLF_TEXT: case CRLF_TEXT:
case CRLF_AUTO: case CRLF_AUTO:
/* fall through */ /* fall through */
@ -254,17 +256,15 @@ static int crlf_to_git(const char *path, const char *src, size_t len,
if (crlf_action == CRLF_AUTO || crlf_action == CRLF_AUTO_INPUT || crlf_action == CRLF_AUTO_CRLF) { if (crlf_action == CRLF_AUTO || crlf_action == CRLF_AUTO_INPUT || crlf_action == CRLF_AUTO_CRLF) {
if (convert_is_binary(len, &stats)) if (convert_is_binary(len, &stats))
return 0; return 0;
/*
if (crlf_action == CRLF_AUTO_INPUT || crlf_action == CRLF_AUTO_CRLF) { * If the file in the index has any CR in it, do not convert.
/* * This is the new safer autocrlf handling.
* If the file in the index has any CR in it, do not convert. */
* This is the new safer autocrlf handling. if (checksafe == SAFE_CRLF_RENORMALIZE)
*/ checksafe = SAFE_CRLF_FALSE;
if (has_cr_in_index(path)) else if (has_cr_in_index(path))
return 0; return 0;
}
} }
check_safe_crlf(path, crlf_action, &stats, checksafe); check_safe_crlf(path, crlf_action, &stats, checksafe);
/* Optimization: No CRLF? Nothing to convert, regardless. */ /* Optimization: No CRLF? Nothing to convert, regardless. */
@ -320,12 +320,10 @@ static int crlf_to_worktree(const char *path, const char *src, size_t len,
return 0; return 0;
if (crlf_action == CRLF_AUTO || crlf_action == CRLF_AUTO_INPUT || crlf_action == CRLF_AUTO_CRLF) { if (crlf_action == CRLF_AUTO || crlf_action == CRLF_AUTO_INPUT || crlf_action == CRLF_AUTO_CRLF) {
if (crlf_action == CRLF_AUTO_INPUT || crlf_action == CRLF_AUTO_CRLF) { /* If we have any CR or CRLF line endings, we do not touch it */
/* If we have any CR or CRLF line endings, we do not touch it */ /* This is the new safer autocrlf-handling */
/* This is the new safer autocrlf-handling */ if (stats.lonecr || stats.crlf )
if (stats.lonecr || stats.crlf ) return 0;
return 0;
}
if (convert_is_binary(len, &stats)) if (convert_is_binary(len, &stats))
return 0; return 0;
@ -786,7 +784,11 @@ static void convert_attrs(struct conv_attrs *ca, const char *path)
ca->drv = git_path_check_convert(ccheck + 2); ca->drv = git_path_check_convert(ccheck + 2);
if (ca->crlf_action != CRLF_BINARY) { if (ca->crlf_action != CRLF_BINARY) {
enum eol eol_attr = git_path_check_eol(ccheck + 3); enum eol eol_attr = git_path_check_eol(ccheck + 3);
if (eol_attr == EOL_LF) if (ca->crlf_action == CRLF_AUTO && eol_attr == EOL_LF)
ca->crlf_action = CRLF_AUTO_INPUT;
else if (ca->crlf_action == CRLF_AUTO && eol_attr == EOL_CRLF)
ca->crlf_action = CRLF_AUTO_CRLF;
else if (eol_attr == EOL_LF)
ca->crlf_action = CRLF_TEXT_INPUT; ca->crlf_action = CRLF_TEXT_INPUT;
else if (eol_attr == EOL_CRLF) else if (eol_attr == EOL_CRLF)
ca->crlf_action = CRLF_TEXT_CRLF; ca->crlf_action = CRLF_TEXT_CRLF;
@ -845,9 +847,9 @@ const char *get_convert_attr_ascii(const char *path)
case CRLF_AUTO: case CRLF_AUTO:
return "text=auto"; return "text=auto";
case CRLF_AUTO_CRLF: case CRLF_AUTO_CRLF:
return "text=auto eol=crlf"; /* This is not supported yet */ return "text=auto eol=crlf";
case CRLF_AUTO_INPUT: case CRLF_AUTO_INPUT:
return "text=auto eol=lf"; /* This is not supported yet */ return "text=auto eol=lf";
} }
return ""; return "";
} }
@ -949,7 +951,7 @@ int renormalize_buffer(const char *path, const char *src, size_t len, struct str
src = dst->buf; src = dst->buf;
len = dst->len; len = dst->len;
} }
return ret | convert_to_git(path, src, len, dst, SAFE_CRLF_FALSE); return ret | convert_to_git(path, src, len, dst, SAFE_CRLF_RENORMALIZE);
} }
/***************************************************************** /*****************************************************************

View File

@ -7,7 +7,8 @@
enum safe_crlf { enum safe_crlf {
SAFE_CRLF_FALSE = 0, SAFE_CRLF_FALSE = 0,
SAFE_CRLF_FAIL = 1, SAFE_CRLF_FAIL = 1,
SAFE_CRLF_WARN = 2 SAFE_CRLF_WARN = 2,
SAFE_CRLF_RENORMALIZE = 3
}; };
extern enum safe_crlf safe_crlf; extern enum safe_crlf safe_crlf;

View File

@ -114,7 +114,7 @@ test_expect_success 'autocrlf=true does not normalize CRLF files' '
test -z "$LFonlydiff" -a -z "$CRLFonlydiff" -a -z "$LFwithNULdiff" test -z "$LFonlydiff" -a -z "$CRLFonlydiff" -a -z "$LFwithNULdiff"
' '
test_expect_success 'text=auto, autocrlf=true _does_ normalize CRLF files' ' test_expect_success 'text=auto, autocrlf=true does not normalize CRLF files' '
rm -f .gitattributes tmp LFonly CRLFonly LFwithNUL && rm -f .gitattributes tmp LFonly CRLFonly LFwithNUL &&
git config core.autocrlf true && git config core.autocrlf true &&
@ -126,7 +126,7 @@ test_expect_success 'text=auto, autocrlf=true _does_ normalize CRLF files' '
LFonlydiff=$(git diff LFonly) && LFonlydiff=$(git diff LFonly) &&
CRLFonlydiff=$(git diff CRLFonly) && CRLFonlydiff=$(git diff CRLFonly) &&
LFwithNULdiff=$(git diff LFwithNUL) && LFwithNULdiff=$(git diff LFwithNUL) &&
test -z "$LFonlydiff" -a -n "$CRLFonlydiff" -a -z "$LFwithNULdiff" test -z "$LFonlydiff" -a -z "$CRLFonlydiff" -a -z "$LFwithNULdiff"
' '
test_expect_success 'text=auto, autocrlf=true does not normalize binary files' ' test_expect_success 'text=auto, autocrlf=true does not normalize binary files' '

View File

@ -175,8 +175,8 @@ attr_ascii () {
text,lf) echo "text eol=lf" ;; text,lf) echo "text eol=lf" ;;
text,crlf) echo "text eol=crlf" ;; text,crlf) echo "text eol=crlf" ;;
auto,) echo "text=auto" ;; auto,) echo "text=auto" ;;
auto,lf) echo "text eol=lf" ;; auto,lf) echo "text=auto eol=lf" ;;
auto,crlf) echo "text eol=crlf" ;; auto,crlf) echo "text=auto eol=crlf" ;;
lf,) echo "text eol=lf" ;; lf,) echo "text eol=lf" ;;
crlf,) echo "text eol=crlf" ;; crlf,) echo "text eol=crlf" ;;
,) echo "" ;; ,) echo "" ;;
@ -397,10 +397,9 @@ commit_chk_wrnNNO "" "" false "" "" "" ""
commit_chk_wrnNNO "" "" true LF_CRLF "" "" "" "" commit_chk_wrnNNO "" "" true LF_CRLF "" "" "" ""
commit_chk_wrnNNO "" "" input "" "" "" "" "" commit_chk_wrnNNO "" "" input "" "" "" "" ""
commit_chk_wrnNNO "auto" "" false "$WILC" "$WICL" "$WAMIX" "" "" commit_chk_wrnNNO "auto" "" false "$WILC" "" "" "" ""
commit_chk_wrnNNO "auto" "" true LF_CRLF "" LF_CRLF "" "" commit_chk_wrnNNO "auto" "" true LF_CRLF "" "" "" ""
commit_chk_wrnNNO "auto" "" input "" CRLF_LF CRLF_LF "" "" commit_chk_wrnNNO "auto" "" input "" "" "" "" ""
for crlf in true false input for crlf in true false input
do do
commit_chk_wrnNNO -text "" $crlf "" "" "" "" "" commit_chk_wrnNNO -text "" $crlf "" "" "" "" ""
@ -408,8 +407,8 @@ do
commit_chk_wrnNNO -text crlf $crlf "" "" "" "" "" commit_chk_wrnNNO -text crlf $crlf "" "" "" "" ""
commit_chk_wrnNNO "" lf $crlf "" CRLF_LF CRLF_LF "" CRLF_LF commit_chk_wrnNNO "" lf $crlf "" CRLF_LF CRLF_LF "" CRLF_LF
commit_chk_wrnNNO "" crlf $crlf LF_CRLF "" LF_CRLF LF_CRLF "" commit_chk_wrnNNO "" crlf $crlf LF_CRLF "" LF_CRLF LF_CRLF ""
commit_chk_wrnNNO auto lf $crlf "" CRLF_LF CRLF_LF "" CRLF_LF commit_chk_wrnNNO auto lf $crlf "" "" "" "" ""
commit_chk_wrnNNO auto crlf $crlf LF_CRLF "" LF_CRLF LF_CRLF "" commit_chk_wrnNNO auto crlf $crlf LF_CRLF "" "" "" ""
commit_chk_wrnNNO text lf $crlf "" CRLF_LF CRLF_LF "" CRLF_LF commit_chk_wrnNNO text lf $crlf "" CRLF_LF CRLF_LF "" CRLF_LF
commit_chk_wrnNNO text crlf $crlf LF_CRLF "" LF_CRLF LF_CRLF "" commit_chk_wrnNNO text crlf $crlf LF_CRLF "" LF_CRLF LF_CRLF ""
done done
@ -454,9 +453,9 @@ do
check_in_repo_NNO -text "" $crlf LF CRLF CRLF_mix_LF LF_mix_CR CRLF_nul check_in_repo_NNO -text "" $crlf LF CRLF CRLF_mix_LF LF_mix_CR CRLF_nul
check_in_repo_NNO -text lf $crlf LF CRLF CRLF_mix_LF LF_mix_CR CRLF_nul check_in_repo_NNO -text lf $crlf LF CRLF CRLF_mix_LF LF_mix_CR CRLF_nul
check_in_repo_NNO -text crlf $crlf LF CRLF CRLF_mix_LF LF_mix_CR CRLF_nul check_in_repo_NNO -text crlf $crlf LF CRLF CRLF_mix_LF LF_mix_CR CRLF_nul
check_in_repo_NNO auto "" $crlf LF LF LF LF_mix_CR CRLF_nul check_in_repo_NNO auto "" $crlf LF CRLF CRLF_mix_LF LF_mix_CR CRLF_nul
check_in_repo_NNO auto lf $crlf LF LF LF LF_mix_CR LF_nul check_in_repo_NNO auto lf $crlf LF CRLF CRLF_mix_LF LF_mix_CR CRLF_nul
check_in_repo_NNO auto crlf $crlf LF LF LF LF_mix_CR LF_nul check_in_repo_NNO auto crlf $crlf LF CRLF CRLF_mix_LF LF_mix_CR CRLF_nul
check_in_repo_NNO text "" $crlf LF LF LF LF_mix_CR LF_nul check_in_repo_NNO text "" $crlf LF LF LF LF_mix_CR LF_nul
check_in_repo_NNO text lf $crlf LF LF LF LF_mix_CR LF_nul check_in_repo_NNO text lf $crlf LF LF LF LF_mix_CR LF_nul
check_in_repo_NNO text crlf $crlf LF LF LF LF_mix_CR LF_nul check_in_repo_NNO text crlf $crlf LF LF LF LF_mix_CR LF_nul
@ -509,7 +508,7 @@ do
checkout_files text "$id" "crlf" "$crlf" "$ceol" CRLF CRLF CRLF CRLF_mix_CR CRLF_nul checkout_files text "$id" "crlf" "$crlf" "$ceol" CRLF CRLF CRLF CRLF_mix_CR CRLF_nul
# currently the same as text, eol=XXX # currently the same as text, eol=XXX
checkout_files auto "$id" "lf" "$crlf" "$ceol" LF CRLF CRLF_mix_LF LF_mix_CR LF_nul checkout_files auto "$id" "lf" "$crlf" "$ceol" LF CRLF CRLF_mix_LF LF_mix_CR LF_nul
checkout_files auto "$id" "crlf" "$crlf" "$ceol" CRLF CRLF CRLF CRLF_mix_CR CRLF_nul checkout_files auto "$id" "crlf" "$crlf" "$ceol" CRLF CRLF CRLF_mix_LF LF_mix_CR LF_nul
done done
# core.autocrlf false, different core.eol # core.autocrlf false, different core.eol
@ -517,7 +516,7 @@ do
# core.autocrlf true # core.autocrlf true
checkout_files "" "$id" "" true "$ceol" CRLF CRLF CRLF_mix_LF LF_mix_CR LF_nul checkout_files "" "$id" "" true "$ceol" CRLF CRLF CRLF_mix_LF LF_mix_CR LF_nul
# text: core.autocrlf = true overrides core.eol # text: core.autocrlf = true overrides core.eol
checkout_files auto "$id" "" true "$ceol" CRLF CRLF CRLF LF_mix_CR LF_nul checkout_files auto "$id" "" true "$ceol" CRLF CRLF CRLF_mix_LF LF_mix_CR LF_nul
checkout_files text "$id" "" true "$ceol" CRLF CRLF CRLF CRLF_mix_CR CRLF_nul checkout_files text "$id" "" true "$ceol" CRLF CRLF CRLF CRLF_mix_CR CRLF_nul
# text: core.autocrlf = input overrides core.eol # text: core.autocrlf = input overrides core.eol
checkout_files text "$id" "" input "$ceol" LF CRLF CRLF_mix_LF LF_mix_CR LF_nul checkout_files text "$id" "" input "$ceol" LF CRLF CRLF_mix_LF LF_mix_CR LF_nul
@ -531,8 +530,8 @@ do
checkout_files text "$id" "" false "" $NL CRLF $MIX_CRLF_LF $MIX_LF_CR $LFNUL checkout_files text "$id" "" false "" $NL CRLF $MIX_CRLF_LF $MIX_LF_CR $LFNUL
checkout_files text "$id" "" false native $NL CRLF $MIX_CRLF_LF $MIX_LF_CR $LFNUL checkout_files text "$id" "" false native $NL CRLF $MIX_CRLF_LF $MIX_LF_CR $LFNUL
# auto: core.autocrlf=false and core.eol unset(or native) uses native eol # auto: core.autocrlf=false and core.eol unset(or native) uses native eol
checkout_files auto "$id" "" false "" $NL CRLF $MIX_CRLF_LF LF_mix_CR LF_nul checkout_files auto "$id" "" false "" $NL CRLF CRLF_mix_LF LF_mix_CR LF_nul
checkout_files auto "$id" "" false native $NL CRLF $MIX_CRLF_LF LF_mix_CR LF_nul checkout_files auto "$id" "" false native $NL CRLF CRLF_mix_LF LF_mix_CR LF_nul
done done
# Should be the last test case: remove some files from the worktree # Should be the last test case: remove some files from the worktree

View File

@ -16,6 +16,13 @@ test_description='CRLF merge conflict across text=auto change
test_have_prereq SED_STRIPS_CR && SED_OPTIONS=-b test_have_prereq SED_STRIPS_CR && SED_OPTIONS=-b
compare_files () {
tr '\015\000' QN <"$1" >"$1".expect &&
tr '\015\000' QN <"$2" >"$2".actual &&
test_cmp "$1".expect "$2".actual &&
rm "$1".expect "$2".actual
}
test_expect_success setup ' test_expect_success setup '
git config core.autocrlf false && git config core.autocrlf false &&
@ -30,7 +37,7 @@ test_expect_success setup '
git branch side && git branch side &&
echo "* text=auto" >.gitattributes && echo "* text=auto" >.gitattributes &&
touch file && echo first line >file &&
git add .gitattributes file && git add .gitattributes file &&
test_tick && test_tick &&
git commit -m "normalize file" && git commit -m "normalize file" &&
@ -81,7 +88,7 @@ test_expect_success 'Merge after setting text=auto' '
rm -f .gitattributes && rm -f .gitattributes &&
git reset --hard a && git reset --hard a &&
git merge b && git merge b &&
test_cmp expected file compare_files expected file
' '
test_expect_success 'Merge addition of text=auto' ' test_expect_success 'Merge addition of text=auto' '
@ -99,7 +106,7 @@ test_expect_success 'Merge addition of text=auto' '
rm -f .gitattributes && rm -f .gitattributes &&
git reset --hard b && git reset --hard b &&
git merge a && git merge a &&
test_cmp expected file compare_files expected file
' '
test_expect_success 'Detect CRLF/LF conflict after setting text=auto' ' test_expect_success 'Detect CRLF/LF conflict after setting text=auto' '
@ -121,7 +128,7 @@ test_expect_success 'Detect CRLF/LF conflict after setting text=auto' '
git reset --hard a && git reset --hard a &&
test_must_fail git merge b && test_must_fail git merge b &&
fuzz_conflict file >file.fuzzy && fuzz_conflict file >file.fuzzy &&
test_cmp expected file.fuzzy compare_files expected file.fuzzy
' '
test_expect_success 'Detect LF/CRLF conflict from addition of text=auto' ' test_expect_success 'Detect LF/CRLF conflict from addition of text=auto' '
@ -143,7 +150,7 @@ test_expect_success 'Detect LF/CRLF conflict from addition of text=auto' '
git reset --hard b && git reset --hard b &&
test_must_fail git merge a && test_must_fail git merge a &&
fuzz_conflict file >file.fuzzy && fuzz_conflict file >file.fuzzy &&
test_cmp expected file.fuzzy compare_files expected file.fuzzy
' '
test_expect_failure 'checkout -m after setting text=auto' ' test_expect_failure 'checkout -m after setting text=auto' '
@ -158,7 +165,7 @@ test_expect_failure 'checkout -m after setting text=auto' '
git reset --hard initial && git reset --hard initial &&
git checkout a -- . && git checkout a -- . &&
git checkout -m b && git checkout -m b &&
test_cmp expected file compare_files expected file
' '
test_expect_failure 'checkout -m addition of text=auto' ' test_expect_failure 'checkout -m addition of text=auto' '
@ -173,7 +180,7 @@ test_expect_failure 'checkout -m addition of text=auto' '
git reset --hard initial && git reset --hard initial &&
git checkout b -- . && git checkout b -- . &&
git checkout -m a && git checkout -m a &&
test_cmp expected file compare_files expected file
' '
test_expect_failure 'cherry-pick patch from after text=auto was added' ' test_expect_failure 'cherry-pick patch from after text=auto was added' '
@ -187,7 +194,7 @@ test_expect_failure 'cherry-pick patch from after text=auto was added' '
git reset --hard b && git reset --hard b &&
test_must_fail git cherry-pick a >err 2>&1 && test_must_fail git cherry-pick a >err 2>&1 &&
grep "[Nn]othing added" err && grep "[Nn]othing added" err &&
test_cmp expected file compare_files expected file
' '
test_expect_success 'Test delete/normalize conflict' ' test_expect_success 'Test delete/normalize conflict' '