我有一个git结帐。所有的文件权限都不同于git认为它们应该是什么,因此它们都显示为修改。

没有触及文件的内容(只是想修改权限),我如何设置所有文件的权限,git认为他们应该是什么?


当前回答

我知道这很古老,但我来自谷歌,我没有找到答案

如果你没有零钱想留着,我有一个简单的解决办法:

git config core.fileMode true
git reset --hard HEAD

其他回答

你也可以尝试一个前/后结帐挂钩。

参见:自定义Git - Git钩子

最简单的方法就是把权限改回来。正如@kroger指出的,git只跟踪可执行位。所以你可能只需要运行chmod -x文件名来修复它(或者如果需要的话+x)。

谢谢@muhqu的精彩回答。在我的例子中,并不是所有的更改文件都更改了权限,这阻止了命令的工作。

$ git diff -p -R --no-ext-diff --no-color | grep -E "^(diff|(old|new) mode)" --color=never
diff --git b/file1 a/file1
diff --git b/file2 a/file2
old mode 100755
new mode 100644
$ git diff -p -R --no-ext-diff --no-color | grep -E "^(diff|(old|new) mode)" --color=never | git apply
warning: file1 has type 100644, expected 100755

然后补丁将停止,文件将保持不变。

如果有人有类似的问题,我通过调整命令来解决这个问题,只有权限更改文件:

grep -E "^old mode (100644|100755)" -B1 -A1

或者git别名

git config --global --add alias.permission-reset '!git diff -p -R --no-ext-diff --no-color | grep -E "^old mode (100644|100755)" -B1 -A1 --color=never | git apply'

muhqu的答案中使用的Git diff -p可能不能显示所有的差异。

我在Cygwin看到了这个我不拥有的文件 模式改变完全被忽略。filemode为false(这是MSysGit的默认值)

这段代码直接读取元数据:

(set -o errexit pipefail nounset;
git ls-tree HEAD -z | while read -r -d $'\0' mask type blob path
do
    if [ "$type" != "blob" ]; then continue; fi;
    case "$mask" in
    #do not touch other bits
    100644) chmod a-x "$path";;
    100755) chmod a+x "$path";;
    *) echo "invalid: $mask $type $blob\t$path" >&2; false;;
    esac
done)

一个非生产级的一行代码(完全替换了掩码):

git ls-tree HEAD | perl -ne '/^10(0\d{3}) blob \S+\t(.+)$/ && { system "chmod",$1,$2 || die }'

(“$ \0”转至http://transnum.blogspot.ru/2008/11/bashs-read-built-in-supports-0-as.html)

Git会跟踪文件权限,并在使用Git diff -p创建补丁时公开权限更改。所以我们只需要:

创建反向补丁 只包括权限更改 将补丁应用到我们的工作副本

一句话:

git diff -p -R --no-ext-diff --no-color \
    | grep -E "^(diff|(old|new) mode)" --color=never  \
    | git apply

你也可以把它作为一个别名添加到你的git配置…

git config --global --add alias.permission-reset '!git diff -p -R --no-ext-diff --no-color | grep -E "^(diff|(old|new) mode)" --color=never | git apply'

...你可以通过以下方式调用它:

git permission-reset

注意,如果shell是bash,请确保在!git周围使用'而不是'引号,否则它将被您运行的最后一个git命令替换。

感谢@Mixologic指出,只要在git diff上使用-R,就不再需要繁琐的sed命令了。