有些脚本在检查更改时不能正确工作。
我是这样试的:
VN=$(git describe --abbrev=7 HEAD 2>/dev/null)
git update-index -q --refresh
CHANGED=$(git diff-index --name-only HEAD --)
if [ ! -z $CHANGED ];
then VN="$VN-mod"
fi
是否存在某种布尔检查,自上次提交以来是否有更改,或者我如何真正测试本地存储库是否有新的更改?
我所做的所有这些都是为了一个版本创建脚本(我在这里的某个地方找到的)。
您所做的几乎可以工作:如果它是空的,您应该引用$CHANGED,而-z测试为空,这意味着没有更改。你的意思是:
if [ -n "$CHANGED" ]; then
VN="$VN-mod"
fi
引用Git的Git - version - gen:
git update-index -q --refresh
test -z "$(git diff-index --name-only HEAD --)" ||
VN="$VN-dirty"
看起来你是在抄袭,但你忘记了引用的细节。
当然,你也可以这样做:
if git diff-index --quiet HEAD --; then
# No changes
else
# Changes
fi
或者如果你只关心“某事发生了变化”的情况:
if ! git diff-index --quiet HEAD --; then
VN="$VN-mod"
fi
使用——quiet的好处是Git可以在遇到单个差异时立即停止处理,因此它可能不需要检查整个工作树。
下面是一组很棒的Bash脚本函数,它可以检查是否存在差异,将其打印给用户,并提示用户是否希望在部署之前提交更改。它是为Heroku和Python应用程序构建的,但对于任何其他应用程序,它几乎不需要更改。
commit(){
echo "Please enter a commit message..."
read msg
git add . --all
git commit -am $msg
}
check_commit(){
echo ========== CHECKING FOR CHANGES ========
changes=$(git diff)
if [ -n "$changes" ]; then
echo ""
echo "*** CHANGES FOUND ***"
echo "$changes"
echo ""
echo "You have uncomitted changes."
echo "Would you like to commit them (y/n)?"
read n
case $n in
"y") commit;;
"n") echo "Changes will not be included...";;
*) echo "invalid option";;
esac
else
echo "... No changes found"
fi
}
deploy(){
check_commit
echo ========== DEPLOYING TO HEROKU ========
git push heroku master
heroku run python manage.py syncdb
}
你可以从gist上复制:https://gist.github.com/sshadmand/f33afe7c9071bb725105
OP的问题已经超过9年了。我不知道那个时候的男人是怎么说的,但现在是这样说的:
--porcelain[=<version>]
Give the output in an easy-to-parse format for scripts. This is similar to the
short output, but will remain stable across Git versions and regardless of user
configuration. See below for details.
The version parameter is used to specify the format version. This is optional and
defaults to the original version v1 format.
这表明,“瓷器”论点非常适合用于测试回购的变化状态。
Wrt OP的问题,“是否有某种布尔检查自上次提交以来是否有更改,或者我如何真正测试本地存储库是否有新的更改?”
我不认为bash本身有布尔数据类型,但这可能足够接近:
[ -z "`git status --porcelain`" ] && echo "NULL-NO DIFFS" || echo "DIFFS EXIST"
这可以作为脚本的if-then-else形式重新转换,或者在git repo文件夹中按原样在CLI中执行。否则,使用-C选项和目标repo的路径规范:
git -C ~/path/to/MyGitRepo status --porcelain
附录:
有些人建议使用-u,——untracked-file选项来避免报告想要忽略的文件的状态。注意,这带来了一个不幸的副作用:新添加的文件也没有处于状态。该选项在某些情况下是有用的,但在使用前要仔细考虑。
我查阅了很多Stack Overflow的答案,但没有一个符合我的期望。也就是说,如果出现以下情况,脚本会出错:
当前本地状态与原始/主文件之间有任何差异(包括:对现有文件的更改;新建未提交和已提交的文件,以及文件删除),或
Master有尚未推送到origin/ Master的提交。
最后的脚本可以这样使用:
if ! (git diff --exit-code origin/master..master > /dev/null) \
|| ! (git diff --exit-code master > /dev/null) \
|| ! [[ -z "$(git status --porcelain)" ]] ; then
echo "Your local repo has some changes that aren't pushed to origin/master ."
exit 1
fi
解释:
Git状态——瓷器将呈现尚未提交的文件列表。用[[-z "$(…)"如果命令返回了一个非空字符串,]]将会出错。
Git diff——exit-code master > /dev/null如果你当前提交的回购状态与主分支不同将会出错。
Git diff——exit-code origin/master..Master > /dev/null如果您当前的主分支与原点/主分支不同,将会出错。
这是有用的,例如,如果你想确保你的回购可能是安全的删除/清除。
为什么这样更好呢?
答案只依赖于git diff -exit-code…不会错误,如果你有新的未提交的非存储文件;
仅依赖于git状态的答案——porcelain不会告诉您有尚未推送到远程的提交。