Git 中糟糕的想法

作者: Corey Quinn

本文改编自一个每年都被 GitHub Universe 忠实拒绝的演讲。我不明白为什么......

不管好坏,git 已经成为开源社区最普遍的工具之一。它可以帮助你有效地管理代码。它可以帮助远隔千里的工程师们互相协作。其核心非常简单,这就是为什么这么多博客文章中的图表不可避免地看起来都像图 1 中所示的那样。

Image removed.

图 1. Git 模型 (来源: https://nvie.com)

鲜少被详细讨论的不幸真相是 git 有其阴暗面:它会让我们觉得自己很蠢。我不在乎你是谁——我们都会遇到一个时刻,耸耸肩,放弃,然后赶紧去 Stack Overflow(座右铭:“此主题已关闭,因为它与主题无关”)上查找最佳方法,以摆脱我们自己造成的糟糕境地。唯一的问题是你会在疯狂吞噬你之前,或者在你开始靠养山羊为生之前,能在这条兔子洞里走多远。

究其核心,git 所做的就是跟踪文件和文件夹的更改。git commit 有效地在给定时间点拍摄文件系统的快照(以添加到暂存区的项目为代表)


cquinn@1d732dc08938 ~/demo1 % git init
Initialized empty Git repository in /home/cquinn/demo1/.git/
cquinn@1d732dc08938(master|...) ~/demo1 % git add ubuntu.iso
cquinn@1d732dc08938(master|·1) ~/demo1 % git commit
 ↪-m "Initial commit"
[master (root-commit) b0d3bfb] Initial commit
 1 file changed, 0 insertions(+), 0 deletions(-)
 create mode 100644 ubuntu.iso
cquinn@1d732dc08938(master|✓) ~/demo1 % git rm --cached
 ↪ubuntu.iso
rm 'ubuntu.iso'
cquinn@1d732dc08938(master|·1✓) ~/demo1 % git
 ↪commit -m "There I fixed it"
[master 2d86934] There I fixed it
 1 file changed, 0 insertions(+), 0 deletions(-)
 delete mode 100644 ubuntu.iso
cquinn@1d732dc08938(master|...) ~/demo1 % du -hs .git
174M    .git

因此,如果你做了一些愚蠢的事情,例如提交大型二进制文件,你不能仅仅回滚提交——它仍然会存在于你的 git 仓库中。如果你已经将该内容推送到其他地方,你就需要强制重写历史记录,可以使用 git-filter-branchbfg。无论哪种方式,这都会给与你共享仓库的其他人带来额外的不便。

从根本上说,git 所做的就是在仓库的顶层创建一个 .git 文件夹。这个子目录包含随时间变化的文件和文件夹。等等,难道没有工具可以做到这一点吗?


cquinn@1d732dc08938 ~/demo2 % git init
Initialized empty Git repository in /home/cquinn/demo2/.git/
cquinn@1d732dc08938(master|✓) ~/demo2 % cd .git
cquinn@1d732dc08938 ~/demo2/.git % ls
HEAD  branches  config  description  hooks  info  objects refs
cquinn@1d732dc08938 ~/demo2/.git % git init
Initialized empty Git repository in /home/cquinn/demo2/
↪.git/.git/

我不确定你 为什么要做这样的事情,但重点是你肯定可以这样做。


cquinn@1d732dc08938 ~/demo2 % git

你有没有试过开始输入一个 git 命令,然后在谷歌搜索时迷失了方向?然后你找到了命令并粘贴进去


cquinn@1d732dc08938 ~/demo2 % git git status git: 'git' is not
a git command. See 'git --help'.

Did you mean this?
    ign

然后你觉得自己很蠢。让我们解决这个问题


# echo git @? \> /usr/local/bin/git-git
    cquinn@1d732dc08938(master|✓) ~/demo2 % sudo bash
    root@1d732dc08938:~/demo2# echo 'git $@' >
     ↪/usr/local/bin/git-git
    root@1d732dc08938:~/demo2# chmod +x /usr/local/bin/git-git

好了,就这样


cquinn@1d732dc08938(master|&#check;) ~/demo2 % git git git git
 ↪git git git status
On branch master

Initial commit

nothing to commit (create/copy files and use "git add" to track)

我不是说这是一个好主意——只是说它可以做到。

真正有用的 Git 技巧

曾经在语法上搞砸过吗?


cquinn@1d732dc08938(master|·1) ~/demo4 % git stts
git: 'stts' is not a git command. See 'git --help'.

Did you mean this?
    status

然后你坐在那里为自己感到难过。欢呼吧:git 具有自动纠错设置


cquinn@1d732dc08938(master|·1) ~/demo4 % git config
 ↪--global help.autocorrect 8
cquinn@1d732dc08938(master|·1) ~/demo4 % git stts
WARNING: You called a Git command named 'stts',
 ↪which does not exist.
Continuing under the assumption that you meant 'status'
in 0.8 seconds automatically...
On branch master
Your branch is up-to-date with 'origin/master'.

Changes to be committed:
   (use "git reset HEAD <file>..." to unstage)

         new file:   file2.txt

--global 标签将此应用于用户的 ~/.gitconfig,但更相关的部分是最后两个参数。help.autocorrect 启用自动错别字检测,而 8 是你必须在几十分之一秒内按下 Ctrl-C 的时间,之后 git 才会对你的环境做一些可怕的事情。

我想指出的最后一个工具是 myrepos。此工具允许你有效地操作构成你公司糟糕的微服务架构的众多仓库(有趣的事实:微服务最初是在一次会议演讲中作为一个玩笑提出的,但人们却当真了)


cquinn@Quinnversion ~ % mr list         31384 10:48:52
 ↪Fri 06- 8-2018
mr list: /Users/cquinn/.config/vcsh/repo.d/gitconfig.git

mr list: /Users/cquinn/.config/vcsh/repo.d/mr.git

mr list: /Users/cquinn/.config/vcsh/repo.d/ssh.git

mr list: /Users/cquinn/.config/vcsh/repo.d/tmux.git

mr list: /Users/cquinn/.config/vcsh/repo.d/vim.git

mr list: /Users/cquinn/.config/vcsh/repo.d/zsh.git

mr list: /Users/cquinn/Dropbox/src/docPR/complete/
↪amazon-cloud-directory-developer-guide

mr list: finished (7 ok)
cquinn@Quinnversion ~ % mr status       31385 10:48:53
 ↪Fri 06- 8-2018
mr status: /Users/cquinn/.config/vcsh/repo.d/gitconfig.git
On branch master
Your branch is up to date with 'origin/master'.

Changes not staged for commit:
  (use "git add <file>..." to update what will be committed)
  (use "git checkout -- <file>..." to discard changes in
    ↪working directory)

        modified:   ../../../../.gitconfig

no changes added to commit (use "git add" and/or
 ↪"git commit -a")

mr status: /Users/cquinn/.config/vcsh/repo.d/mr.git
On branch master
Your branch is up to date with 'origin/master'.

Changes not staged for commit:
   (use "git add <file>..." to update what will be committed)
   (use "git checkout -- <file>..." to discard changes in
     ↪working directory)

         modified:   ../../../../.mrconfig

no changes added to commit (use "git add" and/or
 ↪"git commit -a")

mr status: /Users/cquinn/.config/vcsh/repo.d/ssh.git

mr status: /Users/cquinn/.config/vcsh/repo.d/tmux.git
On branch master
Your branch is up to date with 'origin/master'.

nothing to commit, working tree clean

mr status: /Users/cquinn/.config/vcsh/repo.d/vim.git

mr status: /Users/cquinn/.config/vcsh/repo.d/zsh.git
Behind origin/master by        1 commits
 M ../../../../.zsh/functions/gitstatus.py
 M ../../../../.zshrc

mr status: /Users/cquinn/Dropbox/src/docPR/complete/
↪amazon-cloud-directory-developer-guide

mr status: finished (7 ok)

它还接受并发参数来并行化工作负载;例如,mr -j8 status 会产生八个工作线程。

还有两个临别提示给你。首先,如果你正在使用 GitHub,请安装 Hub 作为 git 的包装器;它扩展了命令以包含诸如 forks、pull requests 和 pages 等概念。其次,设置你的提示符(有很多项目可以在各种 shells 中工作;可以自己找找或者在 Twitter 上问我 以获得最新的建议)来反映你当前的 git 状态——你所在的哪个分支,是否有未提交的工作等等。细微的视觉提示将帮助你避免犯可怕的错误。

Corey 是 The Duckbill Group 的云经济学家。Corey 专注于通过使公司的 AWS 账单更小且不那么可怕来帮助他们改进;主持 Screaming in the Cloud 和 AWS Morning Brief 播客;并策划 Last Week in AWS,这是一份每周新闻通讯,总结了 AWS 新闻、博客和工具的最新信息,并带有一些讽刺意味。

加载 Disqus 评论