小编典典

检查 Git 中是否需要拉取

all

如何检查远程存储库是否已更改并且我需要拉取?

现在我使用这个简单的脚本:

git pull --dry-run | grep -q -v 'Already up-to-date.' && changed=1

但是比较重。

有没有更好的办法?理想的解决方案是检查所有远程分支,并返回更改分支的名称和每个分支中新提交的数量。


阅读 109

收藏
2022-03-03

共1个答案

小编典典

首先使用git remote update,
更新您的远程参考。然后你可以做几件事之一,例如:

  1. git status -uno将告诉您您正在跟踪的分支是领先、落后还是已经发散。如果它什么也没说,本地和远程是一样的。

  2. git show-branch *master将向您显示名称以“master”结尾的所有分支中的提交(例如 masterorigin/master )。

如果你使用-vwith git remote update( git remote -v update)
你可以看到哪些分支得到了更新,所以你真的不需要任何进一步的命令。

但是,您似乎想在脚本或程序中执行此操作,并最终得到一个真/假值。 如果是这样,有一些方法可以检查您当前的HEAD
提交和您正在跟踪的分支的负责人之间的关系,尽管由于有四种可能的结果,您不能将其简化为是/否的答案。但是,如果你准备做一个,pull --rebase那么你可以将“本地落后”和“本地已经分歧”视为“需要拉动”,将其他两个视为“不需要拉动”。

您可以使用 获取任何 ref 的提交 ID git rev-parse <ref>,因此您可以对 masterorigin/master
执行此操作并比较它们。如果它们相等,则分支是相同的。如果它们不相等,您想知道哪个领先于另一个。Usinggit merge-base master origin/master会告诉你两个分支的共同祖先,如果它们没有分歧,这将与一个或另一个相同。如果你得到三个不同的 id,则分支已经分道扬镳。

要正确执行此操作,例如在脚本中,您需要能够引用当前分支以及它正在跟踪的远程分支。中的 bash
提示设置功能/etc/bash_completion.d有一些有用的代码来获取分支名称。但是,您可能实际上并不需要获取名称。Git
有一些简洁的简写方式来引用分支和提交(如 参考资料中所述git rev-parse --help)。特别是,您可以@用于当前分支(假设您不处于分离头状态)@{u}及其上游分支(例如origin/master)。Sogit merge-base @ @{u}将返回当前分支及其上游分支的提交(哈希),git rev-parse @并将git rev-parse @{u}为您提供两个提示的哈希。这可以总结为以下脚本:

#!/bin/sh

UPSTREAM=${1:-'@{u}'}
LOCAL=$(git rev-parse @)
REMOTE=$(git rev-parse "$UPSTREAM")
BASE=$(git merge-base @ "$UPSTREAM")

if [ $LOCAL = $REMOTE ]; then
    echo "Up-to-date"
elif [ $LOCAL = $BASE ]; then
    echo "Need to pull"
elif [ $REMOTE = $BASE ]; then
    echo "Need to push"
else
    echo "Diverged"
fi

注意: 旧版本的 git 不允许单独@使用,因此您可能必须@{0}改用。

该行UPSTREAM=${1:-'@{u}'}允许您选择性地显式传递上游分支,以防您要检查与为当前分支配置的远程分支不同的远程分支。这通常是
remotename/branchname 的形式。如果没有给出参数,则默认值为@{u}

该脚本假定您已经完成了 a git fetchor git remote update,以使跟踪分支保持最新。我没有将其构建到脚本中,因为能够将获取和比较作为单独的操作进行更灵活,例如,如果您想在不获取的情况下进行比较,因为您最近已经获取了。

2022-03-03