整理下常见的 git 问题。
refusing to merge
完整报错为 refusing to merge unrelated histories
信息,通过 pull
时会报上面的错误,可能性比较大的是远端仓库的历史被修改了,例如通过 --amend
重新提交。此时可以通过 --allow-unrelated-history
选项尝试解决,如果仍然报错 Not possible to fast-forward, aborting.
可以尝试如下方案。
----- 获取远端提交
$ git fetch --all
----- 强制覆盖本地分支,最后的参数为要拉取的远端分支名
$ git reset --hard origin/master
实际上 pull
命令是 fetch
和 merge
命令的组合,也就是先从远端仓库抓取内容,然后立即尝试将其合并进当前所在分支。
合并策略
这是一个在 Git 2.27 添加的 hint 提示,详细内容如下:
hint: Pulling without specifying how to reconcile divergent branches is
hint: discouraged. You can squelch this message by running one of the following
hint: commands sometime before your next pull:
hint:
hint: git config pull.rebase false # merge (the default strategy)
hint: git config pull.rebase true # rebase
hint: git config pull.ff only # fast-forward only
hint:
hint: You can replace "git config" with "git config --global" to set a default
hint: preference for all repositories. You can also pass --rebase, --no-rebase,
hint: or --ff-only on the command line to override the configured default per
hint: invocation.
使用 git pull
命令默认等价于 git fetch
以及 git merge FETCH_HEAD
命令,因为执行了合并动作,通常就需要创建一个 commit
请求,可以通过 git pull --ff-only
或者 git config --global pull.ff only
只执行 fast-forwarded
合并,这种模式不会创建新的提交。
Certificate Issuer
完整的报错是 Peer's Certificate issuer is not recognized.
,主要是由于 HTTPS 的证书无法识别导致,可以将无法识别的证书添加到根证书,或者直接忽略。
这里简单使用后者。
----- 可以通过环境变量设置忽略SSL校验
$ echo 'export GIT_SSL_NO_VERIFY=true' >> ~/.bashrc
$ source ~/.bashrc
----- 也可以在git中进行配置
$ git config http.sslVerify "false"
$ git config --global http.sslVerify false
$ git config http.sslCAinfo /your/path/to/cacert-client.pem
Push.default
对于 2.0 以后的版本,默认安装之后会有如下的警告信息。
warning: push.default is unset; its implicit value is changing in
Git 2.0 from 'matching' to 'simple'. To squelch this message
and maintain the current behavior after the default changes, use:
git config --global push.default matching
To squelch this message and adopt the new behavior now, use:
git config --global push.default simple
See 'git help config' and search for 'push.default' for further information.
(the 'simple' mode was introduced in Git 1.7.11. Use the similar mode
'current' instead of 'simple' if you sometimes use older versions of Git)
在 git 中 push 代码时,会有些固定的策略:
- nothing 什么都不干 (要它干嘛! 测试用的?)。
- matching 本地所有的分支都push上去。
- upstream 当本地分支有upstream (也就是有对应的远程分支) 时 Push 到对应的远程分支。
- simple 和 upstream 一样,但不允许将本地分支提交到远程不一样名字的分支。
- current 把当前的分支 push 到远程的同名分支.
git 1.x
的默认策略是 matching,在 git 2.0
之后 simple 会成为新的默认策略。
matching 不友好之处在于,我们大部分情况都是同步本地的当前分支到远程,如果本地有二三十个分支,那么就会看到大量的刷屏,而且如果其它分支也有更新的话就可能会出现大量的 push fail
信息。
相比来说,simple 这个选项是非常安全的选项,至少能阻止新手误操作覆盖远程分支,所以 git 会在 2.0 时将其作为默认策略。但是需要保证远程与本地的分支名称相同。
大部分情况想要做的只是 push 当前的分支,那么最适合的就是 upstream 。
可以通过如下方式查看、修改全局的配置,相关的配置文件保存在 ~/.gitconfig
中。
$ git config --global --list
$ git config --global push.default upstream
agent refused operation
完整的报错如下。
sign_and_send_pubkey: signing failed: agent refused operation
一般是在服务器添加完公钥后报错,可以通过执行如下命令方式解决。
$ eval "$(ssh-agent -s)"
$ ssh-add
HEAD detached at
git 中的 HEAD 可以理解为一个指针,保存在 .git/HEAD
文件中,如果 git checkout <BranchName>
那么就指向该分支的最新一次本地提交,那么 HEAD
文件的内容会是 ref: ref/heads/develop
;当通过 git checkout <CommitID>
指定某个提交时,就会处于 Detach 状态,文件的内容就成了对应的 CommitID
。
连接超时
从 github.com
同步代码可能会比较慢。
----- 先验证SSH连接
$ ssh -v github.com
... ...
expecting SSH2_MSG_KEX_ECDH_REPLY
----- 指定MAC算法
$ ssh -o MACs=hmac-sha2-256 -v github.com
----- 或者修改配置文件
$ cat ~/.ssh/config
Host github
Hostname github.com
User git
IdentityFile ~/.ssh/id_rsa
MACs hmac-sha2-256
其它
快速复制
当提交仓库过大时,可以在 clone
时通过 --depth=1
指定拉取最新提交,如果后续要获取全部提交,那么可以通过 git fetch --unshallow
即可。
PreCommit
在提交之前可以执行一些操作检查,比如代码打包、代码检测,称之为钩子;如果执行成功继续提交,否则失败就阻止提交。
----- 直接复制示例代码
$ cp .git/hooks/pre-commit.sample .git/hooks/pre-commit
代理
常见的可以参考 toolwa.com/github 以及 www.ghproxy.cn ,推荐使用后者,同时还有多个备选域名映射,当前包括了 github.site、github.store 域名。