Pro Git 中文版


Pro Git Scott Chacon* 2010-07-02 *This is the PDF file for the Pro Git book contents. It is licensed under the Creative Commons Attribution-Non Commercial-Share Alike 3.0 license. I hope you enjoy it, I hope it helps you learn Git, and I hope you’ll support Apress and me by purchasing a print copy of the book at Amazon: http://tinyurl.com/amazonprogit 目录 1 起步 1 1.1 关于版本控制 ............................ 1 1.1.1 本地版本控制系统 ...................... 1 1.1.2 集中化的版本控制系统 .................... 1 1.1.3 分布式版本控制系统 ..................... 2 1.2 Git 的历史 ............................. 3 1.3 Git 基础要点 ............................ 4 1.3.1 直接快照,而非比较差异 ................... 4 1.3.2 近乎所有操作都可本地执行 .................. 5 1.3.3 时刻保持数据完整性 ..................... 5 1.3.4 多数操作仅添加数据 ..................... 5 1.3.5 三种状态 .......................... 6 1.4 安装 Git .............................. 7 1.4.1 从源代码安装 ........................ 7 1.4.2 在 Linux 上安装 ....................... 7 1.4.3 在 Mac 上安装 ........................ 8 1.4.4 在 Windows 上安装 ...................... 8 1.5 初次运行 Git 前的配置 ....................... 9 1.5.1 用户信息 .......................... 9 1.5.2 文本编辑器 ......................... 9 1.5.3 差异分析工具 ........................ 10 1.5.4 查看配置信息 ........................ 10 1.6 获取帮助 .............................. 10 1.7 小结 ................................ 11 2 Git 基础 13 2.1 取得项目的 Git 仓库 ........................ 13 2.1.1 从当前目录初始化 ...................... 13 2.1.2 从现有仓库克隆 ....................... 14 2.2 记录每次更新到仓库 ......................... 14 2.2.1 检查当前文件状态 ...................... 14 2.2.2 跟踪新文件 ......................... 15 2.2.3 暂存已修改文件 ....................... 16 2.2.4 忽略某些文件 ........................ 17 2.2.5 查看已暂存和未暂存的更新 .................. 18 2.2.6 提交更新 .......................... 20 iii 2.2.7 跳过使用暂存区域 ...................... 21 2.2.8 移除文件 .......................... 22 2.2.9 移动文件 .......................... 23 2.3 查看提交历史 ............................ 24 2.3.1 限制输出长度 ........................ 28 2.3.2 使用图形化工具查阅提交历史 ................. 29 2.4 撤消操作 .............................. 29 2.4.1 修改最后一次提交 ...................... 29 2.4.2 取消已经暂存的文件 ..................... 30 2.4.3 取消对文件的修改 ...................... 31 2.5 远程仓库的使用 ........................... 31 2.5.1 查看当前的远程库 ...................... 32 2.5.2 添加远程仓库 ........................ 32 2.5.3 从远程仓库抓取数据 ..................... 33 2.5.4 推送数据到远程仓库 ..................... 33 2.5.5 查看远程仓库信息 ...................... 34 2.5.6 远程仓库的删除和重命名 ................... 35 2.6 打标签 ............................... 35 2.6.1 列显已有的标签 ....................... 35 2.6.2 新建标签 .......................... 36 2.6.3 含附注的标签 ........................ 36 2.6.4 签署标签 .......................... 37 2.6.5 轻量级标签 ......................... 37 2.6.6 验证标签 .......................... 38 2.6.7 后期加注标签 ........................ 38 2.6.8 分享标签 .......................... 39 2.7 技巧和窍门 ............................. 40 2.7.1 自动完成 .......................... 40 2.7.2 Git 命令别名 ........................ 41 2.8 小结 ................................ 42 3 Git 分支 43 3.1 何谓分支 .............................. 43 3.2 基本的分支与合并 .......................... 47 3.2.1 基本分支 .......................... 48 3.2.2 基本合并 .......................... 51 3.2.3 冲突的合并 ......................... 52 3.3 分支管理 .............................. 54 3.4 分支式工作流程 ........................... 55 3.4.1 长期分支 .......................... 56 3.4.2 特性分支 .......................... 56 3.5 远程分支 .............................. 57 3.5.1 推送 ............................ 59 3.5.2 跟踪分支 .......................... 61 3.5.3 删除远程分支 ........................ 62 3.6 衍合 ................................ 62 iv 3.6.1 衍合基础 .......................... 62 3.6.2 更多有趣的衍合 ....................... 64 3.6.3 衍合的风险 ......................... 66 3.7 小结 ................................ 68 4 服务器上的 Git 69 4.1 协议 ................................ 69 4.1.1 本地协议 .......................... 69 优点 ............................ 70 缺点 ............................ 70 4.1.2 SSH 协议 .......................... 70 优点 ............................ 71 缺点 ............................ 71 4.1.3 Git 协议 .......................... 71 优点 ............................ 71 缺点 ............................ 72 4.1.4 HTTP/S 协议 ......................... 72 优点 ............................ 72 缺点 ............................ 73 4.2 在服务器部署 Git .......................... 73 4.2.1 将纯目录转移到服务器 .................... 73 4.2.2 小型安装 .......................... 74 SSH 连接 .......................... 74 4.3 生成 SSH 公钥 ........................... 75 4.4 架设服务器 ............................. 76 4.5 公共访问 .............................. 78 4.6 网页界面 GitWeb .......................... 79 4.7 权限管理器 Gitosis ......................... 80 4.8 Git 进程 .............................. 84 4.9 Git 托管服务 ............................ 86 4.9.1 GitHub ........................... 87 4.9.2 建立账户 .......................... 87 4.9.3 建立新仓库 ......................... 87 4.9.4 从 Subversion 中导入项目 .................. 89 4.9.5 开始合作 .......................... 90 4.9.6 项目页面 .......................... 91 4.9.7 派生(forking)项目 ..................... 92 4.9.8 GitHub 小节 ......................... 93 4.10 小节 ................................ 93 5 分布式 Git 95 5.1 分布式工作流程 ........................... 95 5.1.1 集中式工作流 ........................ 95 5.1.2 集成管理员工作流 ...................... 96 5.1.3 司令官与副官工作流 ..................... 97 5.2 为项目作贡献 ............................ 97 v 5.2.1 提交指南 .......................... 98 5.2.2 私有的小型团队 ....................... 99 5.2.3 私有团队间协作 ....................... 104 5.2.4 公开的小型项目 ....................... 108 5.2.5 公开的大型项目 ....................... 112 5.2.6 小结 ............................ 114 5.3 项目的管理 ............................. 114 5.3.1 使用特性分支进行工作 .................... 115 5.3.2 采纳来自邮件的补丁 ..................... 115 使用 apply 命令应用补丁 ................... 115 使用 am 命令应用补丁 .................... 116 5.3.3 检出远程分支 ........................ 118 5.3.4 决断代码取舍 ........................ 118 5.3.5 代码集成 .......................... 119 合并流程 .......................... 120 大项目的合并流程 ...................... 122 衍合与挑拣(cherry-pick)的流程 ............... 122 5.3.6 给发行版签名 ........................ 123 5.3.7 生成内部版本号 ....................... 124 5.3.8 准备发布 .......................... 125 5.3.9 制作简报 .......................... 125 5.4 小结 ................................ 126 6 Git 工具 127 6.1 修订版本(Revision)选择 ...................... 127 6.1.1 单个修订版本 ........................ 127 6.1.2 简短的SHA .......................... 127 6.1.3 关于 SHA-1 的简短说明 .................... 128 6.1.4 分支引用 .......................... 129 6.1.5 引用日志里的简称 ...................... 129 6.1.6 祖先引用 .......................... 130 6.1.7 提交范围 .......................... 132 双点 ............................ 132 多点 ............................ 133 三点 ............................ 133 6.2 交互式暂存 ............................. 134 6.2.1 暂存和撤回文件 ....................... 134 6.2.2 暂存补丁 .......................... 136 6.3 储藏(Stashing) .......................... 137 6.3.1 储藏你的工作 ........................ 138 6.3.2 从储藏中创建分支 ...................... 139 6.4 重写历史 .............................. 140 6.4.1 改变最近一次提交 ...................... 140 6.4.2 修改多个提交说明 ...................... 141 6.4.3 重排提交 .......................... 142 6.4.4 压制(Squashing)提交 ..................... 143 vi 6.4.5 拆分提交 .......................... 144 6.4.6 核弹级选项: filter-branch .................. 144 从所有提交中删除一个文件 .................. 144 将一个子目录设置为新的根目录 ................ 145 全局性地更换电子邮件地址 .................. 145 6.5 使用 Git 调试 ........................... 146 6.5.1 文件标注 .......................... 146 6.5.2 二分查找 .......................... 147 6.6 子模块 ............................... 148 6.6.1 子模块初步 ......................... 149 6.6.2 克隆一个带子模块的项目 ................... 151 6.6.3 上层项目 .......................... 153 6.6.4 子模块的问题 ........................ 153 6.7 子树合并 .............................. 154 6.8 总结 ................................ 156 7 自定义 Git 157 7.1 配置 Git .............................. 157 7.1.1 客户端基本配置 ....................... 157 core.editor ......................... 158 commit.template ....................... 158 core.pager ......................... 159 user.signingkey ....................... 159 core.excludesfile ...................... 159 help.autocorrect ...................... 159 7.1.2 Git中的着色 ......................... 160 color.ui .......................... 160 color.* ........................... 160 7.1.3 外部的合并与比较工具 .................... 160 7.1.4 格式化与空白 ........................ 163 core.autocrlf ........................ 163 core.whitespace ....................... 164 7.1.5 服务器端配置 ........................ 164 receive.fsckObjects ..................... 164 receive.denyNonFastForwards ................. 165 receive.denyDeletes ..................... 165 7.2 Git属性 .............................. 165 7.2.1 二进制文件 ......................... 165 识别二进制文件 ....................... 165 比较二进制文件 ....................... 166 7.2.2 关键字扩展 ......................... 168 7.2.3 导出仓库 .......................... 170 export-ignore ........................ 170 export-subst ........................ 170 7.2.4 合并策略 .......................... 171 7.3 Git挂钩 .............................. 171 vii 7.3.1 安装一个挂钩 ........................ 171 7.3.2 客户端挂钩 ......................... 171 提交工作流挂钩 ....................... 171 E-mail工作流挂钩 ...................... 172 其他客户端挂钩 ....................... 172 7.3.3 服务器端挂钩 ........................ 173 pre-receive 和 post-receive ................. 173 update ........................... 173 7.4 Git 强制策略实例 .......................... 173 7.4.1 服务端挂钩 ......................... 173 指定特殊的提交信息格式 ................... 174 实现基于用户的访问权限控制列表(ACL)系统 .......... 175 只允许 Fast-Forward 类型的推送 ............... 177 7.4.2 Client-Side Hooks ...................... 179 7.5 总结 ................................ 182 8 Git 与其他系统 183 8.1 Git 与 Subversion ......................... 183 8.1.1 git svn ........................... 183 8.1.2 初始设定 .......................... 184 8.1.3 入门 ............................ 184 8.1.4 提交到 Subversion ...................... 186 8.1.5 拉取最新进展 ........................ 187 8.1.6 Git 分支问题 ........................ 188 8.1.7 Subversion 分支 ....................... 189 创建新的 SVN 分支 ...................... 189 8.1.8 切换当前分支 ........................ 190 8.1.9 对应 Subversion 的命令 ................... 190 SVN 风格的历史 ....................... 190 SVN 日志 .......................... 191 SVN 服务器信息 ....................... 191 略 Subversion 之所略 .................... 192 8.1.10 Git-Svn 总结 ........................ 192 8.2 迁移到 Git ............................. 192 8.2.1 导入 ............................ 192 8.2.2 Subversion ......................... 193 8.2.3 Perforce .......................... 194 8.2.4 自定导入脚本 ........................ 196 8.3 总结 ................................ 201 9 Git 内部原理 203 9.1 底层命令 (Plumbing) 和高层命令 (Porcelain) ............. 203 9.2 Git 对象 .............................. 204 9.2.1 tree (树) 对象 ....................... 206 9.2.2 commit (提交) 对象 ..................... 208 9.2.3 对象存储 .......................... 210 viii 9.3 Git References ........................... 212 9.3.1 HEAD 标记 .......................... 213 9.3.2 Tags ............................ 214 9.3.3 Remotes ........................... 215 9.4 Packfiles ............................. 216 9.5 The Refspec ............................ 218 9.5.1 推送 Refspec ........................ 220 9.5.2 删除引用 .......................... 221 9.6 传输协议 .............................. 221 9.6.1 哑协议 ........................... 221 9.6.2 智能协议 .......................... 223 上传数据 .......................... 223 下载数据 .......................... 224 9.7 维护及数据恢复 ........................... 225 9.7.1 维护 ............................ 225 9.7.2 数据恢复 .......................... 226 9.7.3 移除对象 .......................... 228 9.8 总结 ................................ 231 ix 第1章 起步 本章介绍开始使用 Git 前的相关知识。我们会先了解一些版本控制工具的历史背景,然 后试着在你的系统上把 Git 跑起来,直到最后配置好,可以开始正常的开发工作。读完本 章,你就会理解为什么 Git 会如此流行,为什么你真的需要使用它。 1.1 关于版本控制 什么是版本控制?我真的需要吗?版本控制是一种记录若干文件内容变化,以便将来查阅 特定版本修订情况的系统。在本书所展示的例子中,我们仅对保存着软件源代码的文本文件 作版本控制管理,而实际上,你可以对任何类型的文件进行版本控制。 如果你是位图形或网页设计师,可能会需要保存某一幅图片或页面布局文件的所有修订版 本。采用版本控制系统(VCS)是个明智的选择。有了它你就可以将某个文件回溯到之前的 状态,甚至将整个项目都回退到过去某个时间点的状态。你可以比较文件的变化细节,查 出是谁最后修改了什么地方从而造成某些怪异问题,又是谁在何时报告了某个功能缺陷, 等等。使用版本控制系统通常还意味着,就算你胡来搞砸了整个项目,把文件改的改,删的 删,你也可以轻松恢复到原先的样子。而由此额外增加的工作量却微乎其微。 1.1.1 本地版本控制系统 许多人习惯用复制整个项目目录的方式来保存不同的版本,或许还会改名加上备份时间以 示区别。这么做唯一的好处就是简单,不过坏处却不少:有时候会混淆所在的工作目录,弄 错了文件丢了数据就没了后退的路。 为了解决这个问题,人们很久以前就开发了许多种本地版本控制系统,大多都是采用某种 简单的数据库来记录文件的历次更新差异(见图 1.1)。 其中最流行的一种叫做 rcs,现今许多计算机系统上都还看得到它的踪影。甚至在流行的 Mac OS X 系统上安装了开发者工具包之后,也可以使用 rcs 命令。它的工作原理基本上就 是保存并管理文件补丁(patch)。文件补丁是一种特定格式的文本文件,记录着对应文件 修订前后的内容变化。所以,根据每次修订后的补丁,rcs 可以通过不断打补丁,计算出各 个版本的文件内容。 1.1.2 集中化的版本控制系统 接下来人们又遇到一个问题,如何让在不同系统上的开发者协同工作?于是,集中化的 版本控制系统( Centralized Version Control Systems,简称 CVCS )应运而生。这类系 1 第1章 起步 Scott Chacon Pro Git 图 1.1: 本地版本控制系统 统,诸如 CVS,Subversion 以及 Perforce 等,都有一个单一的集中管理的服务器,保存 所有文件的修订版本,而协同工作的人们都通过客户端连到这台服务器,取出最新的文件或 者提交更新。多年以来,这已成为版本控制系统的标准做法(见图 1.2)。 图 1.2: 集中化的版本控制系统 这种做法带来了许多好处,特别是相较于老式的本地 VCS 来说。现在,每个人都可以一 定程度上看到项目中的其他人正在做些什么。而管理员也可以轻松掌控每个开发者的权限, 并且管理一个 CVCS 要远比在各个客户端上维护本地数据库轻松容易得多。 事分两面,有好有坏。这么做最显而易见的缺点是中央服务器的单点故障。若是宕机一小 时,那么在这一小时内,谁都无法提交更新,也就无法协同工作。如果中央服务器的磁盘发 生故障,并且没做过备份或者备份得不够及时的话,还会有丢失数据的风险。最坏的情况是 彻底丢失整个项目的所有历史更改记录,被客户端提取出来的某些快照数据除外,但这样的 话依然是个问题,你不能保证所有的数据都已经有人提取出来。本地版本控制系统也存在类 似问题,只要整个项目的历史记录被保存在单一位置,就有丢失所有历史更新信息的风险。 1.1.3 分布式版本控制系统 于是分布式版本控制系统( Distributed Version Control System,简称 DVCS )面世 了。在这类系统中,诸如 Git,Mercurial,Bazaar 还有 Darcs 等,客户端并不只提取最 新版本的文件快照,而是把原始的代码仓库完整地镜像下来。这么一来,任何一处协同工作 2 Scott Chacon Pro Git 1.2节 Git 的历史 用的服务器发生故障,事后都可以用任何一个镜像出来的本地仓库恢复。因为每一次的提取 操作,实际上都是一次对代码仓库的完整备份(见图 1.3)。 图 1.3: 分布式版本控制系统 更进一步,许多这类系统都可以指定和若干不同的远端代码仓库进行交互。籍此,你就可 以在同一个项目中,分别和不同工作小组的人相互协作。你可以根据需要设定不同的协作流 程,比方说层次模型式的工作流,这在以前的集中式系统中是无法实现的。 1.2 Git 的历史 同生活中的许多伟大事件一样,Git 诞生于一个极富纷争大举创新的年代。Linux 内核开 源项目有着为数众广的参与者。绝大多数的 Linux 内核维护工作都花在了提交补丁和保存 归档的繁琐事务上(1991-2002年间)。到 2002 年,整个项目组开始启用分布式版本控制 系统 BitKeeper 来管理和维护代码。 到 2005 年的时候,开发 BitKeeper 的商业公司同 Linux 内核开源社区的合作关系结 束,他们收回了免费使用 BitKeeper 的权力。这就迫使 Linux 开源社区(特别是 Linux 的缔造者 Linus Torvalds )不得不吸取教训,只有开发一套属于自己的版本控制系统才不 至于重蹈覆辙。他们对新的系统订了若干目标: • 速度 • 简单的设计 • 对非线性开发模式的强力支持(允许上千个并行开发的分支) • 完全分布式 • 有能力高效管理类似 Linux 内核一样的超大规模项目(速度和数据量) 3 第1章 起步 Scott Chacon Pro Git 自诞生于 2005 年以来,Git 日臻成熟完善,在高度易用的同时,仍然保留着初期设定的目 标。它的速度飞快,极其适合管理大项目,它还有着令人难以置信的非线性分支管理系统 (见第三章),可以应付各种复杂的项目开发需求。 1.3 Git 基础要点 那么,简单地说,Git 究竟是怎样的一个系统呢?请注意,接下来的内容非常重要,若 是理解了 Git 的思想和基本的工作原理,用起来就会知其所以然,游刃有余。在开始学习 Git 的时候,请不要尝试把各种概念和其他的版本控制系统诸如 Subversion 和 Perforce 等相比拟,否则容易混淆每个操作的实际意义。Git 在保存和处理各种信息的时候,虽然操 作起来的命令形式非常相近,但它与其他版本控制系统的做法颇为不同。理解这些差异将有 助于你准确地使用 Git 提供的各种工具。 1.3.1 直接快照,而非比较差异 Git 和其他版本控制系统的主要差别在于,Git 只关心文件数据的整体是否发生变化,而 大多数其他系统则只关心文件内容的具体差异。这类系统(CVS,Subversion,Perforce,Bazaar 等等)每次记录有哪些文件作了更新,以及都更新了哪些行的什么内容,请看图 1.4。 图 1.4: 其他系统在每个版本中记录着各个文件的具体差异 Git 并不保存这些前后变化的差异数据。实际上,Git 更像是把变化的文件作快照后,记 录在一个微型的文件系统中。每次提交更新时,它会纵览一遍所有文件的指纹信息并对文件 作一快照,然后保存一个指向这次快照的索引。为提高性能,若文件没有变化,Git 不会再 次保存,而只对上次保存的快照作一连接。Git 的工作方式就像图 1.5 所示。 图 1.5: Git 保存每次更新时的文件快照 这是 Git 同其他系统的重要区别。它完全颠覆了传统版本控制的套路,并对各个环节的 实现方式作了新的设计。Git 更像是个小型的文件系统,但它同时还提供了许多以此为基础 的超强工具,而不只是一个简单的 VCS。稍后在第三章讨论 Git 分支管理的时候,我们会 再看看这样的设计究竟会带来哪些好处。 4 Scott Chacon Pro Git 1.3节 Git 基础要点 1.3.2 近乎所有操作都可本地执行 在 Git 中的绝大多数操作都只需要访问本地文件和资源,不用连网。但如果用 CVCS 的 话,差不多所有操作都需要连接网络。因为 Git 在本地磁盘上就保存着所有有关当前项目 的历史更新,所以处理起来速度飞快。 举个例子,如果要浏览项目的历史更新摘要,Git 不用跑到外面的服务器上去取数据回 来,而直接从本地数据库读取后展示给你看。所以任何时候你都可以马上翻阅,无需等待。 如果想要看当前版本的文件和一个月前的版本之间有何差异,Git 会取出一个月前的快照和 当前文件作一次差异运算,而不用请求远程服务器来做这件事,或是把老版本的文件拉到本 地来作比较。 用 CVCS 的话,没有网络或者断开 VPN 你就无法做任何事情。但用 Git 的话,就算你在 飞机或者火车上,都可以非常愉快地频繁提交更新,等到了有网络的时候再上传到远程的 镜像仓库。同样,在回家的路上,不用连接 VPN 你也可以继续工作。换作其他版本控制系 统,这么做几乎不可能,抑或非常麻烦。比如 Perforce,如果不连到服务器,几乎什么都 做不了(译注:实际上手工修改文件权限改为可写之后是可以编辑文件的,只是其他开发者 无法通过 Perforce 知道你正在对此文件进行修改。);如果是 Subversion 或 CVS,虽然 可以编辑文件,但无法提交更新,因为数据库在网络上。看上去好像这些都不是什么大问 题,但在实际体验过之后,你就会惊喜地发现,这其实是会带来很大不同的。 1.3.3 时刻保持数据完整性 在保存到 Git 之前,所有数据都要进行内容的校验和(checksum)计算,并将此结果作 为数据的唯一标识和索引。换句话说,不可能在你修改了文件或目录之后,Git 一无所知。 这项特性作为 Git 的设计哲学,建在整体架构的最底层。所以如果文件在传输时变得不完 整,或者磁盘损坏导致文件数据缺失,Git 都能立即察觉。 Git 使用 SHA-1 算法计算数据的校验和,通过对文件的内容或目录的结构计算出一个 SHA-1 哈希值,作为指纹字符串。该字串由 40 个十六进制字符(0-9 及 a-f)组成,看起 来就像是: 24b9da6552252987aa493b52f8696cd6d3b00373 Git 的工作完全依赖于这类指纹字串,所以你会经常看到这样的哈希值。实际上,所有保 存在 Git 数据库中的东西都是用此哈希值来作索引的,而不是靠文件名。 1.3.4 多数操作仅添加数据 常用的 Git 操作大多仅仅是把数据添加到数据库。因为任何一种不可逆的操作,比如删 除数据,要回退或重现都会非常困难。在别的 VCS 中,若还未提交更新,就有可能丢失或 者混淆一些修改的内容,但在 Git 里,一旦提交快照之后就完全不用担心丢失数据,特别 是在养成了定期推送至其他镜像仓库的习惯的话。 这种高可靠性令我们的开发工作安心不少,尽管去做各种试验性的尝试好了,再怎样也 不会弄丢数据。至于 Git 内部究竟是如何保存和恢复数据的,我们会在第九章的“幕后细 节”部分再作详述。 5 第1章 起步 Scott Chacon Pro Git 1.3.5 三种状态 好,现在请注意,接下来要讲的概念非常重要。对于任何一个文件,在 Git 内都只有三 种状态:已提交(committed),已修改(modified)和已暂存(staged)。已提交表示该 文件已经被安全地保存在本地数据库中了;已修改表示修改了某个文件,但还没有提交保 存;已暂存表示把已修改的文件放在下次提交时要保存的清单中。 由此我们看到 Git 管理项目时,文件流转的三个工作区域:Git 的本地数据目录,工作 目录以及暂存区域。 图 1.6: 工作目录,暂存区域和 git 目录 每个项目都有一个 git 目录,它是 Git 用来保存元数据和对象数据库的地方。该目录非 常重要,每次克隆镜像仓库的时候,实际拷贝的就是这个目录里面的数据。 从项目中取出某个版本的所有文件和目录,用以开始后续工作的叫做工作目录。这些文件 实际上都是从 git 目录中的压缩对象数据库中提取出来的,接下来就可以在工作目录中对 这些文件进行编辑。 所谓的暂存区域只不过是个简单的文件,一般都放在 git 目录中。有时候人们会把这个 文件叫做索引文件,不过标准说法还是叫暂存区域。 基本的 Git 工作流程如下所示: 1. 在工作目录中修改某些文件。 2. 对这些修改了的文件作快照,并保存到暂存区域。 3. 提交更新,将保存在暂存区域的文件快照转储到 git 目录中。 所以,我们可以从文件所处的位置来判断状态:如果是 git 目录中保存着的特定版本文 件,就属于已提交状态;如果作了修改并已放入暂存区域,就属于已暂存状态;如果自上次 取出后,作了修改但还没有放到暂存区域,就是已修改状态。到第二章的时候,我们会进一 步了解个中细节,并学会如何善用这些状态,以及如何跳过暂存环节。 6 Scott Chacon Pro Git 1.4节 安装 Git 1.4 安装 Git 是时候动动手了,不过在此之前得先安装好 Git。有许多安装方式,概括起来主要有两 种,一种是通过编译源代码来安装;另一种是使用为特定平台预编译好的安装包。 1.4.1 从源代码安装 若是条件允许,从源代码安装有很多好处,至少可以安装最新的版本。Git 的每个版本 都在不断尝试改进用户体验,所以能通过源代码自己编译安装最新版本就再好不过了。 有些 Linux 版本自带的安装包更新起来并不及时,所以除非你在用最新的 distro 或者 backports,那么从源代码安装其实该算是最佳选择。 Git 的工作需要调用 curl,zlib,openssl,expat,libiconv 等库的代码,所以需要先 安装这些依赖工具。在有 yum 的系统上(比如 Fedora)或者有 apt-get 的系统上(比如 Debian 体系的),可以用下面的命令安装: $ yum install curl-devel expat-devel gettext-devel \ openssl-devel zlib-devel $ apt-get install curl-devel expat-devel gettext-devel \ openssl-devel zlib-devel 之后,从下面的 Git 官方站点下载最新版本源代码: http://git-scm.com/download 然后编译并安装: $ tar -zxf git-1.6.0.5.tar.gz $ cd git-1.6.0.5 $ make prefix=/usr/local all $ sudo make prefix=/usr/local install 现在已经可以用 git 命令了,用 git 把 Git 项目仓库克隆到本地,以便日后随时更新: $ git clone git://git.kernel.org/pub/scm/git/git.git 1.4.2 在 Linux 上安装 如果要在 Linux 上安装预编译好的 Git 二进制安装包,可以直接用系统提供的包管理工 具。在 Fedora 上用 yum 安装: $ yum install git-core 在 Ubuntu 这类 Debian 体系的系统上,可以用 apt-get 安装: 7 第1章 起步 Scott Chacon Pro Git $ apt-get instal git-core 1.4.3 在 Mac 上安装 在 Mac 上安装 Git 有两种方式。最容易的当属使用图形化的 Git 安装工具,界面如图 1.7,下载地址在: http://code.google.com/p/git-osx-installer 图 1.7: Git OS X 安装工具体 另一种是通过 MacPorts (http://www.macports.org) 安装。如果已经装好了 MacPorts, 用下面的命令安装 Git: $ sudo port install git-core +svn +doc +bash_completion +gitweb 这种方式就不需要再自己安装依赖库了,Macports 会帮你搞定这些麻烦事。一般上面列 出的安装选项已经够用,要是你想用 Git 连接 Subversion 的代码仓库,还可以加上 +svn 选项,具体将在第八章作介绍。 1.4.4 在 Windows 上安装 在 Windows 上安装 Git 同样轻松,有个叫做 msysGit 的项目提供了安装包,可以从 Google Code 的页面上下载安装文件(.exe): http://code.google.com/p/msysgit 完成安装之后,就可以使用命令行的 git 工具(已经自带了 ssh 客户端)了,另外还有 一个图形界面的 Git 项目管理工具。 8 Scott Chacon Pro Git 1.5节 初次运行 Git 前的配置 1.5 初次运行 Git 前的配置 一般在新的系统上,我们都需要先配置下自己的 Git 工作环境。配置工作只需一次,以 后升级时还会沿用现在的配置。当然,如果需要,你随时可以用相同的命令修改已有的配 置。 Git 提供了一个叫做 git config 的工具(译注:实际是 git-config 命令,只不过可以 通过 git 加一个名字来呼叫此命令。),专门用来配置或读取相应的工作环境变量。而正 是由这些环境变量,决定了 Git 在各个环节的具体工作方式和行为。这些变量可以存放在 以下三个不同的地方: •/etc/gitconfig文件:系统中对所有用户都普遍适用的配置。若使用 git config 时 用 --system 选项,读写的就是这个文件。 • ~/.gitconfig文件:用户目录下的配置文件只适用于该用户。若使用 git config 时 用 --global 选项,读写的就是这个文件。 • 当前项目的 git 目录中的配置文件(也就是工作目录中的 .git/config 文件):这 里的配置仅仅针对当前项目有效。每一个级别的配置都会覆盖上层的相同配置,所以 .git/config 里的配置会覆盖 /etc/gitconfig 中的同名变量。 在 Windows 系统上,Git 会找寻用户主目录下的 .gitconfig 文件。主目录即 $HOME 变量 指定的目录,一般都是 C:\Documents and Settings\$USER。此外,Git 还会尝试找寻 / etc/gitconfig 文件,只不过看当初 Git 装在什么目录,就以此作为根目录来定位。 1.5.1 用户信息 第一个要配置的是你个人的用户名称和电子邮件地址。这两条配置很重要,每次 Git 提 交时都会引用这两条信息,说明是谁提交了更新,所以会随更新内容一起被永久纳入历史记 录: $ git config --global user.name "John Doe" $ git config --global user.email johndoe@example.com 如果用了 --global 选项,那么更改的配置文件就是位于你用户主目录下的那个,以后你 所有的项目都会默认使用这里配置的用户信息。如果要在某个特定的项目中使用其他名字或 者电邮,只要去掉 --global 选项重新配置即可,新的设定保存在当前项目的 .git/config 文件里。 1.5.2 文本编辑器 接下来要设置的是默认使用的文本编辑器。Git 需要你输入一些额外消息的时候,会自动 调用一个外部文本编辑器给你用。默认会使用操作系统指定的默认编辑器,一般可能会是 Vi 或者 Vim。如果你有其他偏好,比如 Emacs 的话,可以重新设置: $ git config --global core.editor emacs 9 第1章 起步 Scott Chacon Pro Git 1.5.3 差异分析工具 还有一个比较常用的是,在解决合并冲突时使用哪种差异分析工具。比如要改用 vimdiff 的话: $ git config --global merge.tool vimdiff Git 可以理解 kdiff3,tkdiff,meld,xxdiff,emerge,vimdiff,gvimdiff,ecmerge, 和 opendiff 等合并工具的输出信息。当然,你也可以指定使用自己开发的工具,具体怎么 做可以参阅第七章。 1.5.4 查看配置信息 要检查已有的配置信息,可以使用 git config --list 命令: $ git config --list user.name=Scott Chacon user.email=schacon@gmail.com color.status=auto color.branch=auto color.interactive=auto color.diff=auto ... 有时候会看到重复的变量名,那就说明它们来自不同的配置文件(比如 /etc/gitconfig 和 ~/.gitconfig),不过最终 Git 实际采用的是最后一个。 也可以直接查阅某个环境变量的设定,只要把特定的名字跟在后面即可,像这样: $ git config user.name Scott Chacon 1.6 获取帮助 想了解 Git 的各式工具该怎么用,可以阅读它们的使用帮助,方法有三: $ git help $ git --help $ man git- 比如,要学习 config 命令可以怎么用,运行: $ git help config 10 Scott Chacon Pro Git 1.7节 小结 我们随时都可以浏览这些帮助信息而无需连网。不过,要是你觉得还不够,可以到 Frenode IRC 服务器(irc.freenode.net)上的 #git 或 #github 频道寻求他人帮助。这 两个频道上总有着上百号人,大多都有着丰富的 git 知识,并且乐于助人。 1.7 小结 至此,你该对 Git 有了点基本的认识,包括它和以前你使用的 CVCS 之间的差别。现 在,在你的系统上应该已经装好了 Git,设置了自己的名字和电邮。接下来让我们继续学习 Git 的基础知识。 11 第2章 Git 基础 读完本章你就能上手使用 Git 了。本章将介绍几个最基本的,也是最常用的 Git 命令, 以后绝大多数时间里用到的也就是这几个命令。读完本章,你就能初始化一个新的代码仓 库,做一些适当的配置;开始或停止跟踪某些文件;暂存或提交某些更新。我们还会展示如 何让 Git 忽略某些文件,或是名称符合特定模式的文件;如何既快且容易地撤消犯下的小 错误;如何浏览项目的更新历史,查看某两次更新之间的差异;以及如何从远程仓库拉数据 下来或者推数据上去。 2.1 取得项目的 Git 仓库 有两种取得 Git 项目仓库的方法。第一种是在现存的目录下,通过导入所有文件来创建 新的 Git 仓库。第二种是从已有的 Git 仓库克隆出一个新的镜像仓库来。 2.1.1 从当前目录初始化 要对现有的某个项目开始用 Git 管理,只需到此项目所在的目录,执行: $ git init 初始化后,在当前目录下会出现一个名为 .git 的目录,所有 Git 需要的数据和资源都 存放在这个目录中。不过目前,仅仅是按照既有的结构框架初始化好了里边所有的文件和目 录,但我们还没有开始跟踪管理项目中的任何一个文件。(在第九章我们会详细说明刚才创 建的 .git 目录中究竟有哪些文件,以及都起些什么作用。) 如果当前目录下有几个文件想要纳入版本控制,需要先用 git add 命令告诉 Git 开始对 这些文件进行跟踪,然后提交: $ git add *.c $ git add README $ git commit -m 'initial project version' 稍后我们再逐一解释每条命令的意思。不过现在,你已经得到了一个实际维护着若干文件 的 Git 仓库。 13 第2章 Git 基础 Scott Chacon Pro Git 2.1.2 从现有仓库克隆 如果想对某个开源项目出一份力,可以先把该项目的 Git 仓库复制一份出来,这就需要 用到 git clone 命令。如果你熟悉其他的 VCS 比如 Subversion,你可能已经注意到这里 使用的是 clone 而不是 checkout。这是个非常重要的差别,Git 收取的是项目历史的所有 数据(每一个文件的每一个版本),服务器上有的数据克隆之后本地也都有了。实际上,即 便服务器的磁盘发生故障,用任何一个克隆出来的客户端都可以重建服务器上的仓库,回到 当初克隆时的状态(可能会丢失某些服务器端的挂钩设置,但所有版本的数据仍旧还在,有 关细节请参考第四章)。 克隆仓库的命令格式为 git clone [url]。比如,要克隆 Ruby 语言的 Git 代码仓库 Grit,可以用下面的命令: $ git clone git://github.com/schacon/grit.git 这会在当前目录下创建一个名为 “grit” 的目录,其中内含一个 .git 的目录,并从同 步后的仓库中拉出所有的数据,取出最新版本的文件拷贝。如果进入这个新建的 grit 目 录,你会看到项目中的所有文件已经在里边了,准备好后续的开发和使用。如果希望在克隆 的时候,自己定义要新建的项目目录名称,可以在上面的命令最后指定: $ git clone git://github.com/schacon/grit.git mygrit 唯一的差别就是,现在新建的目录成了 mygrit,其他的都和上边的一样。 Git 支持许多数据传输协议。之前的例子使用的是 git:// 协议,不过你也可以用 http(s):// 或者 user@server:/path.git 表示的 SSH 传输协议。我们会在第四章详细介 绍所有这些协议在服务器端该如何配置使用,以及各种方式之间的利弊。 2.2 记录每次更新到仓库 现在我们手上已经有了一个真实项目的 Git 仓库,并从这个仓库中取出了所有文件的工 作拷贝。接下来,对这些文件作些修改,在完成了一个阶段的目标之后,提交本次更新到仓 库。 请记住,工作目录下面的所有文件都不外乎这两种状态:已跟踪或未跟踪。已跟踪的文件 是指本来就被纳入版本控制管理的文件,在上次快照中有它们的记录,工作一段时间后,它 们的状态可能是未更新,已修改或者已放入暂存区。而所有其他文件都属于未跟踪文件。它 们既没有上次更新时的快照,也不在当前的暂存区域。初次克隆某个仓库时,工作目录中的 所有文件都属于已跟踪文件,且状态为未修改。 在编辑过某些文件之后,Git 将这些文件标为已修改。我们逐步把这些修改过的文件放到 暂存区域,然后等最后一次性提交暂存区域的所有文件更新,如此重复。所以使用 Git 时 的文件状态变化周期如图 2.1 所示。 2.2.1 检查当前文件状态 要确定哪些文件当前处于什么状态,可以用 git status 命令。如果在克隆仓库之后立即 执行此命令,会看到类似这样的输出: 14 Scott Chacon Pro Git 2.2节 记录每次更新到仓库 图 2.1: 文件的状态变化周期 $ git status # On branch master nothing to commit (working directory clean) 这说明你现在的工作目录相当干净。换句话说,当前没有任何跟踪着的文件,也没有任何 文件在上次提交后更改过。此外,上面的信息还表明,当前目录下没有出现任何处于未跟踪 的新文件,否则 Git 会在这里列出来。最后,该命令还显示了当前所在的分支是 master, 这是默认的分支名称,实际是可以修改的,现在不必多虑。下一章我们就会详细讨论分支和 引用。 现在让我们用 vim 编辑一个新文件 README,保存退出后运行 git status 会看到该文件 出现在未跟踪文件列表中: $ vim README $ git status # On branch master # Untracked files: # (use "git add ..." to include in what will be committed) # # README nothing added to commit but untracked files present (use "git add" to track) 就是在“Untracked files”这行下面。Git 不会自动将之纳入跟踪范围,除非你明明白 白地告诉它这么做,因而不用担心把临时文件什么的也归入版本管理。不过现在我们确实想 要跟踪管理 README 这个文件。 2.2.2 跟踪新文件 使用命令 git add 开始跟踪一个新文件。所以,要跟踪 README 文件,运行: $ git add README 15 第2章 Git 基础 Scott Chacon Pro Git 此时再运行 git status 命令,会看到 README 文件已被跟踪,并处于暂存状态: $ git status # On branch master # Changes to be committed: # (use "git reset HEAD ..." to unstage) # # new file: README # 只要在 “Changes to be committed” 这行下面的,就说明是已暂存状态。如果此时提 交,那么该文件此时此刻的版本将被留存在历史记录中。你可能会想起之前我们使用 git init 后就运行了 git add 命令,开始跟踪当前目录下的文件。git add 后可以接要跟踪的 文件或目录的路径。如果是目录的话,就说明要递归跟踪所有该目录下的文件。 2.2.3 暂存已修改文件 现在我们修改下之前已跟踪过的文件 benchmarks.rb,然后再次运行 status 命令,会看 到这样的状态报告: $ git status # On branch master # Changes to be committed: # (use "git reset HEAD ..." to unstage) # # new file: README # # Changed but not updated: # (use "git add ..." to update what will be committed) # # modified: benchmarks.rb # 文件 benchmarks.rb 出现在 “Changed but not updated” 这行下面,说明已跟踪文件 的内容发生了变化,但还没有放到暂存区。要暂存这次更新,需要运行 git add 命令(这 是个多功能命令,根据目标文件的状态不同,此命令的效果也不同:可以用它开始跟踪新文 件,或者把已跟踪的文件放到暂存区,还能用于合并时把有冲突的文件标记为已解决状态 等)。现在让我们运行 git add 将 benchmarks.rb 放到暂存区,然后再看看 git status 的输出: $ git add benchmarks.rb $ git status # On branch master # Changes to be committed: # (use "git reset HEAD ..." to unstage) # # new file: README # modified: benchmarks.rb 16 Scott Chacon Pro Git 2.2节 记录每次更新到仓库 # 现在两个文件都已暂存,下次提交时就会一并记录到仓库。假设此时,你想要在 bench- marks.rb 里再加条注释,重新编辑存盘后,准备好提交。不过且慢,再运行 git status 看看: $ vim benchmarks.rb $ git status # On branch master # Changes to be committed: # (use "git reset HEAD ..." to unstage) # # new file: README # modified: benchmarks.rb # # Changed but not updated: # (use "git add ..." to update what will be committed) # # modified: benchmarks.rb # 见鬼!benchmarks.rb 文件出现了两次!一次算未暂存,一次算已暂存,这怎么可能呢? 好吧,实际上 Git 只不过暂存了你运行 git add 命令时的版本,如果现在提交,那么提交 的是添加注释前的版本,而非当前工作目录中的版本。所以,运行了 git add 之后又作了 修订的文件,需要重新运行 git add 把最新版本重新暂存起来: $ git add benchmarks.rb $ git status # On branch master # Changes to be committed: # (use "git reset HEAD ..." to unstage) # # new file: README # modified: benchmarks.rb # 2.2.4 忽略某些文件 一般我们总会有些文件无需纳入 Git 的管理,也不希望它们总出现在未跟踪文件列表。 通常都是些自动生成的文件,像是日志或者编译过程中创建的等等。我们可以创建一个名为 .gitignore 的文件,列出要忽略的文件模式,来看一个简单的例子: $ cat .gitignore *.[oa] *~ 17 第2章 Git 基础 Scott Chacon Pro Git 第一行告诉 Git 忽略所有以 .o 或 .a 结尾的文件。一般这类对象文件和存档文件都是 编译过程中出现的,我们用不着跟踪它们的版本。第二行告诉 Git 忽略所有以波浪符(~) 结尾的文件,许多文本编辑软件(比如 Emacs)都用这样的文件名保存副本。此外,你可能 还需要忽略 log,tmp 或者 pid 目录,以及自动生成的文档等等。要养成一开始就设置好 .gitignore 文件的习惯,以免将来误提交这类无用的文件。 文件 .gitignore 的格式规范如下: • 所有空行或者以注释符号 # 开头的行都会被 Git 忽略。 • 可以使用标准的 glob 模式匹配。 • 匹配模式最后跟反斜杠(/)说明要忽略的是目录。 • 要忽略指定模式以外的文件或目录,可以在模式前加上惊叹号(!)取反。 所谓的 glob 模式是指 shell 所使用的简化了的正则表达式。星号(*)匹配零个或多个任 意字符;[abc] 匹配任何一个列在方括号中的字符(这个例子要么匹配一个 a,要么匹配一 个 b,要么匹配一个 c);问号(?)只匹配一个任意字符;如果在方括号中使用短划线分 隔两个字符,表示所有在这两个字符范围内的都可以匹配(比如 [0-9] 表示匹配所有 0 到 9 的数字)。 我们再看一个 .gitignore 文件的例子: # 此为注释 – 将被 Git 忽略 *.a # 忽略所有 .a 结尾的文件 !lib.a # 但 lib.a 除外 /TODO # 仅仅忽略项目根目录下的 TODO 文件,不包括 subdir/TODO build/ # 忽略 build/ 目录下的所有文件 doc/*.txt # 会忽略 doc/notes.txt 但不包括 doc/server/arch.txt 2.2.5 查看已暂存和未暂存的更新 实际上 git status 的显示比较简单,仅仅是列出了修改过的文件,如果要查看具体修改 了什么地方,可以用 git diff 命令。稍后我们会详细介绍 git diff,不过现在,它已经 能回答我们的两个问题了:当前作的哪些更新还没有暂存?有哪些更新已经暂存起来准备好 了下次提交? git diff 会使用文件补丁的格式显示具体添加和删除的行。 假如再次修改 README 文件后暂存,然后编辑 benchmarks.rb 文件后先别暂存,运行 status 命令,会看到: $ git status # On branch master # Changes to be committed: # (use "git reset HEAD ..." to unstage) # # new file: README # # Changed but not updated: # (use "git add ..." to update what will be committed) # # modified: benchmarks.rb # 18 Scott Chacon Pro Git 2.2节 记录每次更新到仓库 要查看尚未暂存的文件更新了哪些部分,不加参数直接输入 git diff: $ git diff diff --git a/benchmarks.rb b/benchmarks.rb index 3cb747f..da65585 100644 --- a/benchmarks.rb +++ b/benchmarks.rb @@ -36,6 +36,10 @@ def main @commit.parents[0].parents[0].parents[0] end + run_code(x, 'commits 1') do + git.commits.size + end + run_code(x, 'commits 2') do log = git.commits('master', 15) log.size 此命令比较的是工作目录中当前文件和暂存区域快照之间的差异,也就是修改之后还没有 暂存起来的变化内容。 若要看已经暂存起来的文件和上次提交时的快照之间的差异,可以用 git diff --cached 命令。(Git 1.6.1 及更高版本还允许使用 git diff --staged,效果是相同的,但更好记 些。)来看看实际的效果: $ git diff --cached diff --git a/README b/README new file mode 100644 index 0000000..03902a1 --- /dev/null +++ b/README2 @@ -0,0 +1,5 @@ +grit + by Tom Preston-Werner, Chris Wanstrath + http://github.com/mojombo/grit + +Grit is a Ruby library for extracting information from a Git repository 请注意,单单 git diff 不过是显示还没有暂存起来的改动,而不是这次工作和上次提交 之间的差异。所以有时候你一下子暂存了所有更新过的文件后,运行 git diff 后却什么也 没有,就是这个原因。 像之前说的,暂存 benchmarks.rb 后再编辑,运行 git status 会看到暂存前后的两个 版本: $ git add benchmarks.rb $ echo '# test line' >> benchmarks.rb $ git status # On branch master 19 第2章 Git 基础 Scott Chacon Pro Git # # Changes to be committed: # # modified: benchmarks.rb # # Changed but not updated: # # modified: benchmarks.rb # 现在运行 git diff 看暂存前后的变化: $ git diff diff --git a/benchmarks.rb b/benchmarks.rb index e445e28..86b2f7c 100644 --- a/benchmarks.rb +++ b/benchmarks.rb @@ -127,3 +127,4 @@ end main() ##pp Grit::GitRuby.cache_client.stats +# test line and git diff --cached to see what you’ve staged so far: $ git diff --cached diff --git a/benchmarks.rb b/benchmarks.rb index 3cb747f..e445e28 100644 --- a/benchmarks.rb +++ b/benchmarks.rb @@ -36,6 +36,10 @@ def main @commit.parents[0].parents[0].parents[0] end + run_code(x, 'commits 1') do + git.commits.size + end + run_code(x, 'commits 2') do log = git.commits('master', 15) log.size 2.2.6 提交更新 现在的暂存区域已经准备妥当可以提交了。在此之前,请一定要确认还有什么修改过的 或新建的文件还没有 git add 过,否则提交的时候不会记录这些还没暂存起来的变化。所 以,每次准备提交前,先用 git status 看下,是不是都已暂存起来了,然后再运行提交命 令 git commit: $ git commit 20 Scott Chacon Pro Git 2.2节 记录每次更新到仓库 这种方式会启动文本编辑器以便输入本次提交的说明。(默认会启用 shell 的环境变量 $EDITOR 所指定的软件,一般都是 vim 或 emacs。当然也可以按照第一章介绍的方式,使 用 git config --global core.editor 命令设定你喜欢的编辑软件。) 编辑器会显示类似下面的文本信息(本例选用 Vim 的屏显方式展示): # Please enter the commit message for your changes. Lines starting # with '#' will be ignored, and an empty message aborts the commit. # On branch master # Changes to be committed: # (use "git reset HEAD ..." to unstage) # # new file: README # modified: benchmarks.rb ~ ~ ~ ".git/COMMIT_EDITMSG" 10L, 283C 可以看到,默认的提交消息包含最后一次运行 git status 的输出,放在注释行里,另外 开头还有一空行,供你输入提交说明。你完全可以去掉这些注释行,不过留着也没关系,多 少能帮你回想起这次更新的内容有哪些。(如果觉得这还不够,可以用 -v 选项将修改差异 的每一行都包含到注释中来。)退出编辑器时,Git 会丢掉注释行,将说明内容和本次更新 提交到仓库。 也可以使用 -m 参数后跟提交说明的方式,在一行命令中提交更新: $ git commit -m "Story 182: Fix benchmarks for speed" [master]: created 463dc4f: "Fix benchmarks for speed" 2 files changed, 3 insertions(+), 0 deletions(-) create mode 100644 README 好,现在你已经创建了第一个提交!可以看到,提交后它会告诉你,当前是在哪个分支 (master)提交的,本次提交的完整 SHA-1 校验和是什么(463dc4f),以及在本次提交 中,有多少文件修订过,多少行添改和删改过。 记住,提交时记录的是放在暂存区域的快照,任何还未暂存的仍然保持已修改状态,可以 在下次提交时纳入版本管理。每一次运行提交操作,都是对你项目作一次快照,以后可以回 到这个状态,或者进行比较。 2.2.7 跳过使用暂存区域 尽管使用暂存区域的方式可以精心准备要提交的细节,但有时候这么做略显繁琐。Git 提 供了一个跳过使用暂存区域的方式,只要在提交的时候,给 git commit 加上 -a 选项,Git 就会自动把所有已经跟踪过的文件暂存起来一并提交,从而跳过 git add 步骤: $ git status # On branch master # 21 第2章 Git 基础 Scott Chacon Pro Git # Changed but not updated: # # modified: benchmarks.rb # $ git commit -a -m 'added new benchmarks' [master 83e38c7] added new benchmarks 1 files changed, 5 insertions(+), 0 deletions(-) 看到了吗?提交之前不再需要 git add 文件 benchmarks.rb 了。 2.2.8 移除文件 要从 Git 中移除某个文件,就必须要从已跟踪文件清单中移除(确切地说,是从暂存区 域移除),然后提交。可以用 git rm 命令完成此项工作,并连带从工作目录中删除指定的 文件,这样以后就不会出现在未跟踪文件清单中了。 如果只是简单地从工作目录中手工删除文件,运行 git status 时就会在 “Changed but not updated” 部分(也就是_未暂存_清单)看到: $ rm grit.gemspec $ git status # On branch master # # Changed but not updated: # (use "git add/rm ..." to update what will be committed) # # deleted: grit.gemspec # 然后再运行 git rm 记录此次移除文件的操作: $ git rm grit.gemspec rm 'grit.gemspec' $ git status # On branch master # # Changes to be committed: # (use "git reset HEAD ..." to unstage) # # deleted: grit.gemspec # 最后提交的时候,该文件就不再纳入版本管理了。如果删除之前修改过并且已经放到暂存 区域的话,则必须要用强制删除选项 -f(译注:即 force 的首字母),以防误删除文件后 丢失修改的内容。 另外一种情况是,我们想把文件从 Git 仓库中删除(亦即从暂存区域移除),但仍然希 望保留在当前工作目录中。换句话说,仅是从跟踪清单中删除。比如一些大型日志文件或者 一堆 .a 编译文件,不小心纳入仓库后,要移除跟踪但不删除文件,以便稍后在 .gitignore 文件中补上,用 --cached 选项即可: 22 Scott Chacon Pro Git 2.2节 记录每次更新到仓库 $ git rm --cached readme.txt 后面可以列出文件或者目录的名字,也可以使用 glob 模式。比方说: $ git rm log/\*.log 注意到星号 * 之前的反斜杠 \,因为 Git 有它自己的文件模式扩展匹配方式,所以我们 不用 shell 来帮忙展开(译注:实际上不加反斜杠也可以运行,只不过按照 shell 扩展的 话,仅仅删除指定目录下的文件而不会递归匹配。上面的例子本来就指定了目录,所以效果 等同,但下面的例子就会用递归方式匹配,所以必须加反斜杠。)。此命令删除所有 log/ 目录下扩展名为 .log 的文件。类似的比如: $ git rm \*~ 会递归删除当前目录及其子目录中所有 ~ 结尾的文件。 2.2.9 移动文件 不像其他的 VCS 系统,Git 并不跟踪文件移动操作。如果在 Git 中重命名了某个文件, 仓库中存储的元数据并不会体现出这是一次改名操作。不过 Git 非常聪明,它会推断出究 竟发生了什么,至于具体是如何做到的,我们稍后再谈。 既然如此,当你看到 Git 的 mv 命令时一定会困惑不已。要在 Git 中对文件改名,可以 这么做: $ git mv file_from file_to 它会恰如预期般正常工作。实际上,即便此时查看状态信息,也会明白无误地看到关于重 命名操作的说明: $ git mv README.txt README $ git status # On branch master # Your branch is ahead of 'origin/master' by 1 commit. # # Changes to be committed: # (use "git reset HEAD ..." to unstage) # # renamed: README.txt -> README # 其实,运行 git mv 就相当于运行了下面三条命令: 23 第2章 Git 基础 Scott Chacon Pro Git $ mv README.txt README $ git rm README.txt $ git add README 如此分开操作,Git 也会意识到这是一次改名,所以不管何种方式都一样。当然,直接用 git mv 轻便得多,不过有时候用其他工具批处理改名的话,要记得在提交前删除老的文件 名,再添加新的文件名。 2.3 查看提交历史 在提交了若干更新之后,又或者克隆了某个项目,想回顾下提交历史,可以使用 git log 命令。 接下来的例子会用我专门用于演示的 simplegit 项目,运行下面的命令获取该项目源代 码: git clone git://github.com/schacon/simplegit-progit.git 然后在此项目中运行 git log,应该会看到下面的输出: $ git log commit ca82a6dff817ec66f44342007202690a93763949 Author: Scott Chacon Date: Mon Mar 17 21:52:11 2008 -0700 changed the verison number commit 085bb3bcb608e1e8451d4b2432f8ecbe6306e7e7 Author: Scott Chacon Date: Sat Mar 15 16:40:33 2008 -0700 removed unnecessary test code commit a11bef06a3f659402fe7563abf99ad00de2209e6 Author: Scott Chacon Date: Sat Mar 15 10:31:28 2008 -0700 first commit 默认不用任何参数的话,git log 会按提交时间列出所有的更新,最近的更新排在最上 面。看到了吗,每次更新都有一个 SHA-1 校验和、作者的名字和电子邮件地址、提交时 间,最后缩进一个段落显示提交说明。 git log 有许多选项可以帮助你搜寻感兴趣的提交,接下来我们介绍些最常用的。 我们常用 -p 选项展开显示每次提交的内容差异,用 -2 则仅显示最近的两次更新: 24 Scott Chacon Pro Git 2.3节 查看提交历史 $ git log –p -2 commit ca82a6dff817ec66f44342007202690a93763949 Author: Scott Chacon Date: Mon Mar 17 21:52:11 2008 -0700 changed the verison number diff --git a/Rakefile b/Rakefile index a874b73..8f94139 100644 --- a/Rakefile +++ b/Rakefile @@ -5,7 +5,7 @@ require 'rake/gempackagetask' spec = Gem::Specification.new do |s| - s.version = "0.1.0" + s.version = "0.1.1" s.author = "Scott Chacon" commit 085bb3bcb608e1e8451d4b2432f8ecbe6306e7e7 Author: Scott Chacon Date: Sat Mar 15 16:40:33 2008 -0700 removed unnecessary test code diff --git a/lib/simplegit.rb b/lib/simplegit.rb index a0a60ae..47c6340 100644 --- a/lib/simplegit.rb +++ b/lib/simplegit.rb @@ -18,8 +18,3 @@ class SimpleGit end end - -if $0 == __FILE__ - git = SimpleGit.new - puts git.show -end \ No newline at end of file 在做代码审查,或者要快速浏览其他协作者提交的更新都作了哪些改动时,就可以用这个 选项。此外,还有许多摘要选项可以用,比如 --stat,仅显示简要的增改行数统计: $ git log --stat commit ca82a6dff817ec66f44342007202690a93763949 Author: Scott Chacon Date: Mon Mar 17 21:52:11 2008 -0700 changed the verison number Rakefile | 2 +- 1 files changed, 1 insertions(+), 1 deletions(-) commit 085bb3bcb608e1e8451d4b2432f8ecbe6306e7e7 Author: Scott Chacon 25 第2章 Git 基础 Scott Chacon Pro Git Date: Sat Mar 15 16:40:33 2008 -0700 removed unnecessary test code lib/simplegit.rb | 5 ----- 1 files changed, 0 insertions(+), 5 deletions(-) commit a11bef06a3f659402fe7563abf99ad00de2209e6 Author: Scott Chacon Date: Sat Mar 15 10:31:28 2008 -0700 first commit README | 6 ++++++ Rakefile | 23 +++++++++++++++++++++++ lib/simplegit.rb | 25 +++++++++++++++++++++++++ 3 files changed, 54 insertions(+), 0 deletions(-) 每个提交都列出了修改过的文件,以及其中添加和移除的行数,并在最后列出所有增减行 数小计。 还有个常用的 --pretty 选项,可以指定使用完全不同于默认格式的方式展示提交历 史。比如用 oneline 将每个提交放在一行显示,这在提交数很大时非常有用。另外还有 short,full 和 fuller 可以用,展示的信息或多或少有些不同,请自己动手实践一下看看 效果如何。 $ git log --pretty=oneline ca82a6dff817ec66f44342007202690a93763949 changed the verison number 085bb3bcb608e1e8451d4b2432f8ecbe6306e7e7 removed unnecessary test code a11bef06a3f659402fe7563abf99ad00de2209e6 first commit 但最有意思的是 format,可以定制要显示的记录格式,这样的输出便于后期编程提取分 析,像这样: $ git log --pretty=format:"%h - %an, %ar : %s" ca82a6d - Scott Chacon, 11 months ago : changed the verison number 085bb3b - Scott Chacon, 11 months ago : removed unnecessary test code a11bef0 - Scott Chacon, 11 months ago : first commit 表 2.1 列出了常用的格式占位符写法及其代表的意义。 选项 说明 %H 提交对象(commit)的完整哈希字串 %h 提交对象的简短哈希字串 %T 树对象(tree)的完整哈希字串 %t 树对象的简短哈希字串 %P 父对象(parent)的完整哈希字串 %p 父对象的简短哈希字串 26 Scott Chacon Pro Git 2.3节 查看提交历史 %an 作者(author)的名字 %ae 作者的电子邮件地址 %ad 作者修订日期(可以用 -date= 选项定制格式) %ar 作者修订日期,按多久以前的方式显示 %cn 提交者(committer)的名字 %ce 提交者的电子邮件地址 %cd 提交日期 %cr 提交日期,按多久以前的方式显示 %s 提交说明 你一定奇怪_作者(author)_和_提交者(committer)_之间究竟有何差别,其实作者指 的是实际作出修改的人,提交者指的是最后将此工作成果提交到仓库的人。所以,当你为某 个项目发去补丁,然后某个核心成员将你的补丁并入项目时,你就是作者,而那个核心成员 就是提交者。我们会在第五章再详细介绍两者之间的细致差别。 用 oneline 或 format 时结合 --graph 选项,可以看到开头多出一些 ASCII 字符串表 示的简单图形,形象地展示了每个提交所在的分支及其分化衍合情况。在我们之前提到的 Grit 项目仓库中可以看到: $ git log --pretty=format:"%h %s" --graph * 2d3acf9 ignore errors from SIGCHLD on trap * 5e3ee11 Merge branch 'master' of git://github.com/dustin/grit |\ | * 420eac9 Added a method for getting the current branch. * | 30e367c timeout code and tests * | 5a09431 add timeout protection to grit * | e1193f8 support for heads with slashes in them |/ * d6016bc require time for xmlschema * 11d191e Merge branch 'defunkt' into local 以上只是简单介绍了一些 git log 命令支持的选项。表 2.2 还列出了一些其他常用的选 项及其释义。 选项 说明 -p 按补丁格式显示每个更新之间的差异。 --stat 显示每次更新的文件修改统计信息。 --shortstat 只显示 --stat 中最后的行数修改添加移除统计。 --name-only 仅在提交信息后显示已修改的文件清单。 --name-status 显示新增、修改、删除的文件清单。 --abbrev-commit 仅显示 SHA-1 的前几个字符,而非所有的 40 个字符。 --relative-date 使用较短的相对时间显示(比如,“2 weeks ago”)。 --graph 显示 ASCII 图形表示的分支合并历史。 --pretty 使用其他格式显示历史提交信息。可用的选项包括 oneline,short,full,fuller 和 format(后跟指 定格式)。 27 第2章 Git 基础 Scott Chacon Pro Git 2.3.1 限制输出长度 除了定制输出格式的选项之外,git log 还有许多非常实用的限制输出长度的选项,也就 是只输出部分提交信息。之前我们已经看到过 -2 了,它只显示最近的两条提交,实际上, 这是 - 选项的写法,其中的 n 可以是任何自然数,表示仅显示最近的若干条提交。不过 实践中我们是不太用这个选项的,Git 在输出所有提交时会自动调用分页程序(pager), 要看更早的更新只需翻到下页即可。 另外还有按照时间作限制的选项,比如 --since 和 --until。下面的命令列出所有最近 两周内的提交: $ git log --since=2.weeks 你可以给出各种时间格式,比如说具体的某一天(“2008-01-15”),或者是多久以前 (“2 years 1 day 3 minutes ago”)。 还可以给出若干搜索条件,列出符合的提交。用 --author 选项显示指定作者的提交,用 --grep 选项搜索提交说明中的关键字。(请注意,如果要得到同时满足这两个选项搜索条 件的提交,就必须用 --all-match 选项。) 如果只关心某些文件或者目录的历史提交,可以在 git log 选项的最后指定它们的路 径。因为是放在最后位置上的选项,所以用两个短划线(--)隔开之前的选项和后面限定的 路径名。 表 2.3 还列出了其他常用的类似选项。 选项 说明 -(n) 仅显示最近的 n 条提交 --since, --after 仅显示指定时间之后的提交。 --until, --before 仅显示指定时间之前的提交。 --author 仅显示指定作者相关的提交。 --committer 仅显示指定提交者相关的提交。 来看一个实际的例子,如果要查看 Git 仓库中,2008 年 10 月期间,Junio Hamano 提 交的但未合并的测试脚本(位于项目的 t/ 目录下的文件),可以用下面的查询命令: $ git log --pretty="%h:%s" --author=gitster --since="2008-10-01" \ --before="2008-11-01" --no-merges -- t/ 5610e3b - Fix testcase failure when extended attribute acd3b9e - Enhance hold_lock_file_for_{update,append}() f563754 - demonstrate breakage of detached checkout wi d1a43f2 - reset --hard/read-tree --reset -u: remove un 51a94af - Fix "checkout --track -b newbranch" on detac b0ad11e - pull: allow "git pull origin $something:$cur Git 项目有 20,000 多条提交,但我们给出搜索选项后,仅列出了其中满足条件的 6 条。 28 Scott Chacon Pro Git 2.4节 撤消操作 2.3.2 使用图形化工具查阅提交历史 有时候图形化工具更容易展示历史提交的变化,随 Git 一同发布的 gitk 就是这样一种 工具。它是用 Tcl/Tk 写成的,基本上相当于 git log 命令的可视化版本,凡是 git log 可以用的选项也都能用在 gitk 上。在项目工作目录中输入 gitk 命令后,就会启动图 2.2 所示的界面。 图 2.2: gitk 的图形界面 上半个窗口显示的是历次提交的分支祖先图谱,下半个窗口显示当前点选的提交对应的具 体差异。 2.4 撤消操作 任何时候,你都有可能需要撤消刚才所做的某些操作。接下来,我们会介绍一些基本的撤 消操作相关的命令。请注意,有些操作并不总是可以撤消的,所以请务必谨慎小心,一旦失 误,就有可能丢失部分工作成果。 2.4.1 修改最后一次提交 有时候我们提交完了才发现漏掉了几个文件没有加,或者提交信息写错了。想要撤消刚才 的提交操作,可以使用 --amend 选项重新提交: $ git commit --amend 29 第2章 Git 基础 Scott Chacon Pro Git 此命令将使用当前的暂存区域快照提交。如果刚才提交完没有作任何改动,直接运行此命 令的话,相当于有机会重新编辑提交说明,而所提交的文件快照和之前的一样。 启动文本编辑器后,会看到上次提交时的说明,编辑它确认没问题后保存退出,就会使用 新的提交说明覆盖刚才失误的提交。 如果刚才提交时忘了暂存某些修改,可以先补上暂存操作,然后再运行 --amend 提交: $ git commit -m 'initial commit' $ git add forgotten_file $ git commit --amend 上面的三条命令最终得到一个提交,第二个提交命令修正了第一个的提交内容。 2.4.2 取消已经暂存的文件 接下来的两个小节将演示如何取消暂存区域中的文件,以及如何取消工作目录中已修改的 文件。不用担心,查看文件状态的时候就提示了该如何撤消,所以不需要死记硬背。来看下 面的例子,有两个修改过的文件,我们想要分开提交,但不小心用 git add * 全加到了暂 存区域。该如何撤消暂存其中的一个文件呢?git status 命令的输出会告诉你怎么做: $ git add . $ git status # On branch master # Changes to be committed: # (use "git reset HEAD ..." to unstage) # # modified: README.txt # modified: benchmarks.rb # 就在 “Changes to be committed” 下面,括号中有提示,可以使用 git reset HEAD ... 的方式取消暂存。好吧,我们来试试取消暂存 benchmarks.rb 文件: $ git reset HEAD benchmarks.rb benchmarks.rb: locally modified $ git status # On branch master # Changes to be committed: # (use "git reset HEAD ..." to unstage) # # modified: README.txt # # Changed but not updated: # (use "git add ..." to update what will be committed) # (use "git checkout -- ..." to discard changes in working directory) # # modified: benchmarks.rb # 30 Scott Chacon Pro Git 2.5节 远程仓库的使用 这条命令看起来有些古怪,先别管,能用就行。现在 benchmarks.rb 文件又回到了之前 已修改未暂存的状态。 2.4.3 取消对文件的修改 如果觉得刚才对 benchmarks.rb 的修改完全没有必要,该如何取消修改,回到之前的状 态(也就是修改之前的版本)呢?git status 同样提示了具体的撤消方法,接着上面的例 子,现在未暂存区域看起来像这样: # Changed but not updated: # (use "git add ..." to update what will be committed) # (use "git checkout -- ..." to discard changes in working directory) # # modified: benchmarks.rb # 在第二个括号中,我们看到了抛弃文件修改的命令(至少在 Git 1.6.1 以及更高版本中 会这样提示,如果你还在用老版本,我们强烈建议你升级,以获取最佳的用户体验),让我 们试试看: $ git checkout -- benchmarks.rb $ git status # On branch master # Changes to be committed: # (use "git reset HEAD ..." to unstage) # # modified: README.txt # 可以看到,该文件已经恢复到修改前的版本。你可能已经意识到了,这条命令有些危险, 所有对文件的修改都没有了,因为我们刚刚把之前版本的文件复制过来重写了此文件。所以 在用这条命令前,请务必确定真的不再需要保留刚才的修改。如果只是想回退版本,同时保 留刚才的修改以便将来继续工作,可以用下章介绍的 stashing 和分支来处理,应该会更好 些。 记住,任何已经提交到 Git 的都可以被恢复。即便在已经删除的分支中的提交,或者用 --amend 重新改写的提交,都可以被恢复(关于数据恢复的内容见第九章)。所以,你可能 失去的数据,仅限于没有提交过的,对 Git 来说它们就像从未存在过一样。 2.5 远程仓库的使用 要参与任何一个 Git 项目的协作,必须要了解该如何管理远程仓库。远程仓库是指托管 在网络上的项目仓库,可能会有好多个,其中有些你只能读,另外有些可以写。同他人协作 开发某个项目时,需要管理这些远程仓库,以便推送或拉取数据,分享各自的工作进展。管 理远程仓库的工作,包括添加远程库,移除废弃的远程库,管理各式远程库分支,定义是否 跟踪这些分支,等等。本节我们将详细讨论远程库的管理和使用。 31 第2章 Git 基础 Scott Chacon Pro Git 2.5.1 查看当前的远程库 要查看当前配置有哪些远程仓库,可以用 git remote 命令,它会列出每个远程库的简短 名字。在克隆完某个项目后,至少可以看到一个名为 origin 的远程库,Git 默认使用这个 名字来标识你所克隆的原始仓库: $ git clone git://github.com/schacon/ticgit.git Initialized empty Git repository in /private/tmp/ticgit/.git/ remote: Counting objects: 595, done. remote: Compressing objects: 100% (269/269), done. remote: Total 595 (delta 255), reused 589 (delta 253) Receiving objects: 100% (595/595), 73.31 KiB | 1 KiB/s, done. Resolving deltas: 100% (255/255), done. $ cd ticgit $ git remote origin 也可以加上 -v 选项(译注:此为 —verbose 的简写,取首字母),显示对应的克隆地 址: $ git remote -v origin git://github.com/schacon/ticgit.git 如果有多个远程仓库,此命令将全部列出。比如在我的 Grit 项目中,可以看到: $ cd grit $ git remote -v bakkdoor git://github.com/bakkdoor/grit.git cho45 git://github.com/cho45/grit.git defunkt git://github.com/defunkt/grit.git koke git://github.com/koke/grit.git origin git@github.com:mojombo/grit.git 这样一来,我就可以非常轻松地从这些用户的仓库中,拉取他们的提交到本地。请注意, 上面列出的地址只有 origin 用的是 SSH URL 链接,所以也只有这个仓库我能推送数据上 去(我们会在第四章解释原因)。 2.5.2 添加远程仓库 要添加一个新的远程仓库,可以指定一个简单的名字,以便将来引用,运行 git remote add [shortname] [url]: $ git remote origin $ git remote add pb git://github.com/paulboone/ticgit.git $ git remote -v 32 Scott Chacon Pro Git 2.5节 远程仓库的使用 origin git://github.com/schacon/ticgit.git pb git://github.com/paulboone/ticgit.git 现在可以用字串 pb 指代对应的仓库地址了。比如说,要抓取所有 Paul 有的,但本地仓 库没有的信息,可以运行 git fetch pb: $ git fetch pb remote: Counting objects: 58, done. remote: Compressing objects: 100% (41/41), done. remote: Total 44 (delta 24), reused 1 (delta 0) Unpacking objects: 100% (44/44), done. From git://github.com/paulboone/ticgit * [new branch] master -> pb/master * [new branch] ticgit -> pb/ticgit 现在,Paul 的主干分支(master)已经完全可以在本地访问了,对应的名字是 pb/ master,你可以将它合并到自己的某个分支,或者切换到这个分支,看看有些什么有趣的更 新。 2.5.3 从远程仓库抓取数据 正如之前所看到的,可以用下面的命令从远程仓库抓取数据到本地: $ git fetch [remote-name] 此命令会到远程仓库中拉取所有你本地仓库中还没有的数据。运行完成后,你就可以在本 地访问该远程仓库中的所有分支,将其中某个分支合并到本地,或者只是取出某个分支,一 探究竟。(我们会在第三章详细讨论关于分支的概念和操作。) 如果是克隆了一个仓库,此命令会自动将远程仓库归于 origin 名下。所以,git fetch origin 会抓取从你上次克隆以来别人上传到此远程仓库中的所有更新(或是上次 fetch 以 来别人提交的更新)。有一点很重要,需要记住,fetch 命令只是将远端的数据拉到本地仓 库,并不自动合并到当前工作分支,只有当你确实准备好了,才能手工合并。 如果设置了某个分支用于跟踪某个远端仓库的分支(参见下节及第三章的内容),可以使 用 git pull 命令自动抓取数据下来,然后将远端分支自动合并到本地仓库中当前分支。在 日常工作中我们经常这么用,既快且好。实际上,默认情况下 git clone 命令本质上就是 自动创建了本地的 master 分支用于跟踪远程仓库中的 master 分支(假设远程仓库确实有 master 分支)。所以一般我们运行 git pull,目的都是要从原始克隆的远端仓库中抓取数 据后,合并到工作目录中当前分支。 2.5.4 推送数据到远程仓库 项目进行到一个阶段,要同别人分享目前的成果,可以将本地仓库中的数据推送到远程 仓库。实现这个任务的命令很简单: git push [remote-name] [branch-name]。如果要把 本地的 master 分支推送到 origin 服务器上(再次说明下,克隆操作会自动使用默认的 master 和 origin 名字),可以运行下面的命令: 33 第2章 Git 基础 Scott Chacon Pro Git $ git push origin master 只有在所克隆的服务器上有写权限,或者同一时刻没有其他人在推数据,这条命令才会如 期完成任务。如果在你推数据前,已经有其他人推送了若干更新,那你的推送操作就会被驳 回。你必须先把他们的更新抓取到本地,并到自己的项目中,然后才可以再次推送。有关推 送数据到远程仓库的详细内容见第三章。 2.5.5 查看远程仓库信息 我们可以通过命令 git remote show [remote-name] 查看某个远程仓库的详细信息,比 如要看所克隆的 origin 仓库,可以运行: $ git remote show origin * remote origin URL: git://github.com/schacon/ticgit.git Remote branch merged with 'git pull' while on branch master master Tracked remote branches master ticgit 除了对应的克隆地址外,它还给出了许多额外的信息。它友善地告诉你如果是在 master 分支,就可以用 git pull 命令抓取数据合并到本地。另外还列出了所有处于跟踪状态中的 远端分支。 实际使用过程中,git remote show 给出的信息可能会像这样: $ git remote show origin * remote origin URL: git@github.com:defunkt/github.git Remote branch merged with 'git pull' while on branch issues issues Remote branch merged with 'git pull' while on branch master master New remote branches (next fetch will store in remotes/origin) caching Stale tracking branches (use 'git remote prune') libwalker walker2 Tracked remote branches acl apiv2 dashboard2 issues master postgres Local branch pushed with 'git push' master:master 34 Scott Chacon Pro Git 2.6节 打标签 它告诉我们,运行 git push 时缺省推送的分支是什么(译注:最后两行)。它还显示 了有哪些远端分支还没有同步到本地(译注:第六行的 caching 分支),哪些已同步到 本地的远端分支在远端服务器上已被删除(译注:Stale tracking branches 下面的两个 分支),以及运行 git pull 时将自动合并哪些分支(译注:前四行中列出的 issues 和 master 分支)。 2.5.6 远程仓库的删除和重命名 在新版 Git 中可以用 git remote rename 命令修改某个远程仓库的简短名称,比如想把 pb 改成 paul,可以这么运行: $ git remote rename pb paul $ git remote origin paul 注意,对远程仓库的重命名,也会使对应的分支名称发生变化,原来的 pb/master 分支 现在成了 paul/master。 碰到远端仓库服务器迁移,或者原来的克隆镜像不再使用,又或者某个参与者不再贡献代 码,那么需要移除对应的远端仓库,可以运行 git remote rm 命令: $ git remote rm paul $ git remote origin 2.6 打标签 同大多数 VCS 一样,Git 也可以对某一时间点上的版本打上标签。人们在发布某个软件 版本(比如 v1.0 等等)的时候,经常这么做。本节我们一起来学习如何列出所有可用的标 签,如何新建标签,以及各种不同类型标签之间的差别。 2.6.1 列显已有的标签 列出现有标签的命令非常简单,直接运行 git tag 即可: $ git tag v0.1 v1.3 显示的标签按字母顺序排列,所以标签的先后并不表示重要程度的轻重。 我们可以用特定的搜索模式列出符合条件的标签。在 Git 自身项目仓库中,有着超过 240 个标签,如果你只对 1.4.2 系列的版本感兴趣,可以运行下面的命令: 35 第2章 Git 基础 Scott Chacon Pro Git $ git tag -l 'v1.4.2.*' v1.4.2.1 v1.4.2.2 v1.4.2.3 v1.4.2.4 2.6.2 新建标签 Git 使用的标签有两种类型:轻量级的(lightweight)和含附注的(annotated)。轻量 级标签就像是个不会变化的分支,实际上它就是个指向特定提交对象的引用。而含附注标 签,实际上是存储在仓库中的一个独立对象,它有自身的校验和信息,包含着标签的名字, 电子邮件地址和日期,以及标签说明,标签本身也允许使用 GNU Privacy Guard (GPG) 来 签署或验证。一般我们都建议使用含附注型的标签,以便保留相关信息;当然,如果只是临 时性加注标签,或者不需要旁注额外信息,用轻量级标签也没问题。 2.6.3 含附注的标签 创建一个含附注类型的标签非常简单,用 -a (译注:取 annotated 的首字母)指定标 签名字即可: $ git tag -a v1.4 -m 'my version 1.4' $ git tag v0.1 v1.3 v1.4 而 -m 选项则指定了对应的标签说明,Git 会将此说明一同保存在标签对象中。如果在此 选项后没有给出具体的说明内容,Git 会启动文本编辑软件供你输入。 可以使用 git show 命令查看相应标签的版本信息,并连同显示打标签时的提交对象。 $ git show v1.4 tag v1.4 Tagger: Scott Chacon Date: Mon Feb 9 14:45:11 2009 -0800 my version 1.4 commit 15027957951b64cf874c3557a0f3547bd83b3ff6 Merge: 4a447f7... a6b4c97... Author: Scott Chacon Date: Sun Feb 8 19:02:46 2009 -0800 Merge branch 'experiment' 我们可以看到在提交对象信息上面,列出了此标签的提交者和提交时间,以及相应的标签 说明。 36 Scott Chacon Pro Git 2.6节 打标签 2.6.4 签署标签 如果你有自己的私钥,还可以用 GPG 来签署标签,只需要把之前的 -a 改为 -s (译注: 取 Signed 的首字母)即可: $ git tag -s v1.5 -m 'my signed 1.5 tag' You need a passphrase to unlock the secret key for user: "Scott Chacon " 1024-bit DSA key, ID F721C45A, created 2009-02-09 现在再运行 git show 会看到对应的 GPG 签名也附在其内: $ git show v1.5 tag v1.5 Tagger: Scott Chacon Date: Mon Feb 9 15:22:20 2009 -0800 my signed 1.5 tag -----BEGIN PGP SIGNATURE----- Version: GnuPG v1.4.8 (Darwin) iEYEABECAAYFAkmQurIACgkQON3DxfchxFr5cACeIMN+ZxLKggJQf0QYiQBwgySN Ki0An2JeAVUCAiJ7Ox6ZEtK+NvZAj82/ =WryJ -----END PGP SIGNATURE----- commit 15027957951b64cf874c3557a0f3547bd83b3ff6 Merge: 4a447f7... a6b4c97... Author: Scott Chacon Date: Sun Feb 8 19:02:46 2009 -0800 Merge branch 'experiment' 稍后我们再学习如何验证已经签署的标签。 2.6.5 轻量级标签 轻量级标签实际上就是一个保存着对应提交对象的校验和信息的文件。要创建这样的标 签,一个 -a,-s 或 -m 选项都不用,直接给出标签名字即可: $ git tag v1.4-lw $ git tag v0.1 v1.3 v1.4 v1.4-lw v1.5 现在运行 git show 查看此标签信息,就只有相应的提交对象摘要: 37 第2章 Git 基础 Scott Chacon Pro Git $ git show v1.4-lw commit 15027957951b64cf874c3557a0f3547bd83b3ff6 Merge: 4a447f7... a6b4c97... Author: Scott Chacon Date: Sun Feb 8 19:02:46 2009 -0800 Merge branch 'experiment' 2.6.6 验证标签 可以使用 git tag -v [tag-name] (译注:取 verify 的首字母)的方式验证已经签署的 标签。此命令会调用 GPG 来验证签名,所以你需要有签署者的公钥,存放在 keyring 中, 才能验证: $ git tag -v v1.4.2.1 object 883653babd8ee7ea23e6a5c392bb739348b1eb61 type commit tag v1.4.2.1 tagger Junio C Hamano 1158138501 -0700 GIT 1.4.2.1 Minor fixes since 1.4.2, including git-mv and git-http with alternates. gpg: Signature made Wed Sep 13 02:08:25 2006 PDT using DSA key ID F3119B9A gpg: Good signature from "Junio C Hamano " gpg: aka "[jpeg image of size 1513]" Primary key fingerprint: 3565 2A26 2040 E066 C9A7 4A7D C0C6 D9A4 F311 9B9A 若是没有签署者的公钥,会报告类似下面这样的错误: gpg: Signature made Wed Sep 13 02:08:25 2006 PDT using DSA key ID F3119B9A gpg: Can't check signature: public key not found error: could not verify the tag 'v1.4.2.1' 2.6.7 后期加注标签 你甚至可以在后期对早先的某次提交加注标签。比如在下面展示的提交历史中: $ git log --pretty=oneline 15027957951b64cf874c3557a0f3547bd83b3ff6 Merge branch 'experiment' a6b4c97498bd301d84096da251c98a07c7723e65 beginning write support 0d52aaab4479697da7686c15f77a3d64d9165190 one more thing 6d52a271eda8725415634dd79daabbc4d9b6008e Merge branch 'experiment' 0b7434d86859cc7b8c3d5e1dddfed66ff742fcbc added a commit function 4682c3261057305bdd616e23b64b0857d832627b added a todo file 38 Scott Chacon Pro Git 2.6节 打标签 166ae0c4d3f420721acbb115cc33848dfcc2121a started write support 9fceb02d0ae598e95dc970b74767f19372d61af8 updated rakefile 964f16d36dfccde844893cac5b347e7b3d44abbc commit the todo 8a5cbc430f1a9c3d00faaeffd07798508422908a updated readme 我们忘了在提交 “updated rakefile” 后为此项目打上版本号 v1.2,没关系,现在也 能做。只要在打标签的时候跟上对应提交对象的校验和(或前几位字符)即可: $ git tag -a v1.2 9fceb02 可以看到我们已经补上了标签: $ git tag v0.1 v1.2 v1.3 v1.4 v1.4-lw v1.5 $ git show v1.2 tag v1.2 Tagger: Scott Chacon Date: Mon Feb 9 15:32:16 2009 -0800 version 1.2 commit 9fceb02d0ae598e95dc970b74767f19372d61af8 Author: Magnus Chacon Date: Sun Apr 27 20:43:35 2008 -0700 updated rakefile ... 2.6.8 分享标签 默认情况下,git push 并不会把标签传送到远端服务器上,只有通过显式命令才能分享 标签到远端仓库。其命令格式如同推送分支,运行 git push origin [tagname] 即可: $ git push origin v1.5 Counting objects: 50, done. Compressing objects: 100% (38/38), done. Writing objects: 100% (44/44), 4.56 KiB, done. Total 44 (delta 18), reused 8 (delta 1) To git@github.com:schacon/simplegit.git * [new tag] v1.5 -> v1.5 如果要一次推送所有(本地新增的)标签上去,可以使用 --tags 选项: 39 第2章 Git 基础 Scott Chacon Pro Git $ git push origin --tags Counting objects: 50, done. Compressing objects: 100% (38/38), done. Writing objects: 100% (44/44), 4.56 KiB, done. Total 44 (delta 18), reused 8 (delta 1) To git@github.com:schacon/simplegit.git * [new tag] v0.1 -> v0.1 * [new tag] v1.2 -> v1.2 * [new tag] v1.4 -> v1.4 * [new tag] v1.4-lw -> v1.4-lw * [new tag] v1.5 -> v1.5 现在,其他人克隆共享仓库或拉取数据同步后,也会看到这些标签。 2.7 技巧和窍门 在结束本章之前,我还想和大家分享一些 Git 使用的技巧和窍门。很多使用 Git 的开发 者可能根本就没用过这些技巧,我们也不是说在读过本书后非得用这些技巧不可,但至少应 该有所了解吧。说实话,有了这些小窍门,我们的工作可以变得更简单,更轻松,更高效。 2.7.1 自动完成 如果你用的是 Bash shell,可以试试看 Git 提供的自动完成脚本。下载 Git 的源 代码,进入 contrib/completion 目录,会看到一个 git-completion.bash 文件。将此 文件复制到你自己的用户主目录中(译注:按照下面的示例,还应改名加上点:cp git- completion.bash ∼/.git-completion.bash),并把下面一行内容添加到你的 .bashrc 文 件中: source ~/.git-completion.bash 也可以为系统上所有用户都设置默认使用此脚本。Mac 上将此脚本复制到 /opt/local/ etc/bash_completion.d 目录中,Linux 上则复制到 /etc/bash_completion.d/ 目录中即 可。这两处目录中的脚本,都会在 Bash 启动时自动加载。 如果在 Windows 上安装了 msysGit,默认使用的 Git Bash 就已经配好了这个自动完成 脚本,可以直接使用。 在输入 Git 命令的时候可以敲两次跳格键(Tab),就会看到列出所有匹配的可用命令建 议: $ git co commit config 此例中,键入 git co 然后连按两次 Tab 键,会看到两个相关的建议(命令) commit 和 config。继而输入 m 会自动完成 git commit 命令的输入。 40 Scott Chacon Pro Git 2.7节 技巧和窍门 命令的选项也可以用这种方式自动完成,其实这种情况更实用些。比如运行 git log 的 时候忘了相关选项的名字,可以输入开头的几个字母,然后敲 Tab 键看看有哪些匹配的: $ git log --s --shortstat --since= --src-prefix= --stat --summary 这个技巧不错吧,可以节省很多输入和查阅文档的时间。 2.7.2 Git 命令别名 Git 并不会推断你输入的几个字符将会是哪条命令,不过如果想偷懒,少敲几个命令的字 符,可以用 git config 为命令设置别名。来看看下面的例子: $ git config --global alias.co checkout $ git config --global alias.br branch $ git config --global alias.ci commit $ git config --global alias.st status 现在,如果要输入 git commit 只需键入 git ci 即可。而随着 Git 使用的深入,会有 很多经常要用到的命令,遇到这种情况,不妨建个别名提高效率。 使用这种技术还可以创造出新的命令,比方说取消暂存文件时的输入比较繁琐,可以自己 设置一下: $ git config --global alias.unstage 'reset HEAD --' 这样一来,下面的两条命令完全等同: $ git unstage fileA $ git reset HEAD fileA 显然,使用别名的方式看起来更清楚。另外,我们还经常设置 last 命令: $ git config --global alias.last 'log -1 HEAD' 然后要看最后一次的提交信息,就变得简单多了: $ git last commit 66938dae3329c7aebe598c2246a8e6af90d04646 Author: Josh Goebel Date: Tue Aug 26 19:48:51 2008 +0800 41 第2章 Git 基础 Scott Chacon Pro Git test for current head Signed-off-by: Scott Chacon 可以看出,实际上 Git 只是简单地在命令中替换了你设置的别名。不过有时候我们希望 运行某个外部命令,而非 Git 的附属工具,这个好办,只需要在命令前加上 ! 就行。如果 你自己写了些处理 Git 仓库信息的脚本的话,就可以用这种技术包装起来。作为演示,我 们可以设置用 git visual 启动 gitk: $ git config --global alias.visual "!gitk" 2.8 小结 到目前为止,你已经学会了最基本的 Git 操作:创建和克隆仓库,作出更新,暂存并提 交这些更新,以及查看所有历史更新记录。接下来,我们将学习 Git 的必杀技特性:分支 模型。 42 第3章 Git 分支 几乎每一种版本控制系统都以某种形式支持分支。使用分支意味着你可以从开发主线上分 离开来,然后在不影响主线的同时继续工作。在很多版本控制系统中,这是个昂贵的过程, 常常需要创建一个源代码目录的完整副本,对大型项目来说会花费很长时间。 有人把 Git 的分支模型称为“必杀技特性”,而正是因为它,将 Git 从版本控制系统家 族里区分出来。Git 有何特别之处呢?Git 的分支可谓是难以置信的轻量级,它的新建操作 几乎可以在瞬间完成,并且在不同分支间切换起来也差不多一样快。和许多其他版本控制系 统不同,Git 鼓励在工作流程中频繁使用分支与合并,哪怕一天之内进行许多次都没有关 系。理解分支的概念并熟练运用后,你才会意识到为什么 Git 是一个如此强大而独特的工 具,并从此真正改变你的开发方式。 3.1 何谓分支 为了理解 Git 分支的实现方式,我们需要回顾一下 Git 是如何储存数据的。或许你还记 得第一章的内容,Git 保存的不是文件差异或者变化量,而只是一系列文件快照。 在 Git 中提交时,会保存一个提交(commit)对象,它包含一个指向暂存内容快照的指 针,作者和相关附属信息,以及一定数量(也可能没有)指向该提交对象直接祖先的指针: 第一次提交是没有直接祖先的,普通提交有一个祖先,由两个或多个分支合并产生的提交则 有多个祖先。 为直观起见,我们假设在工作目录中有三个文件,准备将它们暂存后提交。暂存操作会对 每一个文件计算校验和(即第一章中提到的 SHA-1 哈希字串),然后把当前版本的文件快 照保存到 Git 仓库中(Git 使用 blob 类型的对象存储这些快照),并将校验和加入暂存 区域: $ git add README test.rb LICENSE2 $ git commit -m 'initial commit of my project' 当使用 git commit 新建一个提交对象前,Git 会先计算每一个子目录(本例中就是项目 根目录)的校验和,然后在 Git 仓库中将这些目录保存为树(tree)对象。之后 Git 创 建的提交对象,除了包含相关提交信息以外,还包含着指向这个树对象(项目根目录)的指 针,如此它就可以在将来需要的时候,重现此次快照的内容了。 现在,Git 仓库中有五个对象:三个表示文件快照内容的 blob 对象;一个记录着目录树 内容及其中各个文件对应 blob 对象索引的 tree 对象;以及一个包含指向 tree 对象(根 43 第3章 Git 分支 Scott Chacon Pro Git 目录)的索引和其他提交信息元数据的 commit 对象。概念上来说,仓库中的各个对象保存 的数据和相互关系看起来如图 3.1 所示: 图 3.1: 一次提交后仓库里的数据 作些修改后再次提交,那么这次的提交对象会包含一个指向上次提交对象的指针(译注: 即下图中的 parent 对象)。两次提交后,仓库历史会变成图 3.2 的样子: 图 3.2: 多次提交后的 Git 对象数据 现在来谈分支。Git 中的分支,其实本质上仅仅是个指向 commit 对象的可变指针。Git 会使用 master 作为分支的默认名字。在若干次提交后,你其实已经有了一个指向最后一次 提交对象的 master 分支,它在每次提交的时候都会自动向前移动。 图 3.3: 指向提交数据历史的分支 那么,Git 又是如何创建一个新的分支的呢?答案很简单,创建一个新的分支指针。比如 新建一个 testing 分支,可以使用 git branch 命令: $ git branch testing 44 Scott Chacon Pro Git 3.1节 何谓分支 这会在当前 commit 对象上新建一个分支指针(见图 3.4)。 图 3.4: 多个分支指向提交数据的历史 那么,Git 是如何知道你当前在哪个分支上工作的呢?其实答案也很简单,它保存着一 个名为 HEAD 的特别指针。请注意它和你熟知的许多其他版本控制系统(比如 Subversion 或 CVS)里的 HEAD 概念大不相同。在 Git 中,它是一个指向你正在工作中的本地分支的 指针。运行 git branch 命令,仅仅是建立了一个新的分支,但不会自动切换到这个分支中 去,所以在这个例子中,我们依然还在 master 分支里工作(参考图 3.5)。 图 3.5: HEAD 指向当前所在的分支 要切换到其他分支,可以执行 git checkout 命令。我们现在转换到新建的 testing 分 支: $ git checkout testing 这样 HEAD 就指向了 testing 分支(见图3-6)。 这样的实现方式会给我们带来什么好处呢?好吧,现在不妨再提交一次: $ vim test.rb $ git commit -a -m 'made a change' 图 3.7 展示了提交后的结果。 非常有趣,现在 testing 分支向前移动了一格,而 master 分支仍然指向原先 git checkout 时所在的 commit 对象。现在我们回到 master 分支看看: 45 第3章 Git 分支 Scott Chacon Pro Git 图 3.6: HEAD 在你转换分支时指向新的分支 图 3.7: 每次提交后 HEAD 随着分支一起向前移动 $ git checkout master 图 3.8 显示了结果。 图 3.8: HEAD 在一次 checkout 之后移动到了另一个分支 这条命令做了两件事。它把 HEAD 指针移回到 master 分支,并把工作目录中的文件换成 了 master 分支所指向的快照内容。也就是说,现在开始所做的改动,将始于本项目中一个 较老的版本。它的主要作用是将 testing 分支里作出的修改暂时取消,这样你就可以向另 一个方向进行开发。 我们作些修改后再次提交: 46 Scott Chacon Pro Git 3.2节 基本的分支与合并 $ vim test.rb $ git commit -a -m 'made other changes' 现在我们的项目提交历史产生了分叉(如图 3.9 所示),因为刚才我们创建了一个分 支,转换到其中进行了一些工作,然后又回到原来的主分支进行了另外一些工作。这些改变 分别孤立在不同的分支里:我们可以在不同分支里反复切换,并在时机成熟时把它们合并到 一起。而所有这些工作,仅仅需要 branch 和 checkout 这两条命令就可以完成。 图 3.9: 分叉了的分支历史 由于 Git 中的分支实际上仅是一个包含所指对象校验和(40 个字符长度 SHA-1 字串) 的文件,所以创建和销毁一个分支就变得非常廉价。说白了,新建一个分支就是向一个文件 写入 41 个字节(外加一个换行符)那么简单,当然也就很快了。 这和大多数版本控制系统形成了鲜明对比,它们管理分支大多采取备份所有项目文件到特 定目录的方式,所以根据项目文件数量和大小不同,可能花费的时间也会有相当大的差别, 快则几秒,慢则数分钟。而 Git 的实现与项目复杂度无关,它永远可以在几毫秒的时间内 完成分支的创建和切换。同时,因为每次提交时都记录了祖先信息(译注:即 parent 对 象),所以以后要合并分支时,寻找恰当的合并基础(译注:即共同祖先)的工作其实已经 完成了一大半,实现起来非常容易。Git 鼓励开发者频繁使用分支,正是因为有着这些特性 作保障。 接下来看看,我们为什么应该频繁使用分支。 3.2 基本的分支与合并 现在让我们来看一个简单的分支与合并的例子,实际工作中大体也会用到这样的工作流 程: 1. 开发某个网站。 2. 为实现某个新的需求,创建一个分支。 3. 在这个分支上开展工作。 47 第3章 Git 分支 Scott Chacon Pro Git 假设此时,你突然接到一个电话说有个很严重的问题需要紧急修补,那么可以按照下面的方 式处理: 1. 返回到原先已经发布到生产服务器上的分支。 2. 为这次紧急修补建立一个新分支。 3. 测试通过后,将此修补分支合并,再推送到生产服务器上。 4. 切换到之前实现新需求的分支,继续工作。 3.2.1 基本分支 首先,我们假设你正在项目中愉快地工作,并且已经提交了几次更新(见图 3.10)。 图 3.10: 一部分简短的提交历史 现在,你决定要修补问题追踪系统上的 #53 问题。顺带说明下,Git 并不同任何特定的 问题追踪系统打交道。这里为了说明要解决的问题,才把新建的分支取名为 iss53。要新建 并切换到该分支,运行 git checkout 并加上 -b 参数: $ git checkout -b iss53 Switched to a new branch "iss53" 相当于下面这两条命令: $ git branch iss53 $ git checkout iss53 图 3.11 示意该命令的结果。 图 3.11: 创建了一个新的分支指针 接下来,你在网站项目上继续工作并作了一次提交。这会使 iss53 分支的指针随着提交 向前推进,因为它处于检出状态(或者说,你的 HEAD 指针目前正指向它,见图3-12): 48 Scott Chacon Pro Git 3.2节 基本的分支与合并 $ vim index.html $ git commit -a -m 'added a new footer [issue 53]' 图 3.12: iss53 分支随工作进展向前推进 现在你就接到了那个网站问题的紧急电话,需要马上修补。有了 Git ,我们就不需要同 时发布这个补丁和 iss53 里作出的修改,也不需要在创建和发布该补丁到服务器之前花费 很多努力来复原这些修改。唯一需要的仅仅是切换回 master 分支。 不过在此之前,留心你的暂存区或者工作目录里,那些还没有提交的修改,它会和你即将 检出的分支产生冲突从而阻止 Git 为你转换分支。转换分支的时候最好保持一个清洁的工 作区域。稍后会介绍几个绕过这种问题的办法(分别叫做 stashing 和 amending)。目前 已经提交了所有的修改,所以接下来可以正常转换到 master 分支: $ git checkout master Switched to branch "master" 此时工作目录中的内容和你在解决问题 #53 之前一模一样,你可以集中精力进行紧急 修补。这一点值得牢记:Git 会把工作目录的内容恢复为检出某分支时它所指向的那个 commit 的快照。它会自动添加、删除和修改文件以确保目录的内容和你上次提交时完全一 样。 接下来,你得进行紧急修补。我们创建一个紧急修补分支(hotfix)来开展工作,直到搞 定(见图 3.13): $ git checkout -b 'hotfix' Switched to a new branch "hotfix" $ vim index.html $ git commit -a -m 'fixed the broken email address' [hotfix]: created 3a0874c: "fixed the broken email address" 1 files changed, 0 insertions(+), 1 deletions(-) 有必要作些测试,确保修补是成功的,然后把它合并到 master 分支并发布到生产服务 器。用 git merge 命令来进行合并: $ git checkout master $ git merge hotfix 49 第3章 Git 分支 Scott Chacon Pro Git 图 3.13: hotfix 分支是从 master 分支所在点分化出来的 Updating f42c576..3a0874c Fast forward README | 1 - 1 files changed, 0 insertions(+), 1 deletions(-) 请注意,合并时出现了 “Fast forward”(快进)提示。由于当前 master 分支所在的 commit 是要并入的 hotfix 分支的直接上游,Git 只需把指针直接右移。换句话说,如果 顺着一个分支走下去可以到达另一个分支,那么 Git 在合并两者时,只会简单地把指针前 移,因为没有什么分歧需要解决,所以这个过程叫做快进(Fast forward)。 现在的目录变为当前 master 分支指向的 commit 所对应的快照,可以发布了(见图 3.14)。 图 3.14: 合并之后,master 分支和 hotfix 分支指向同一位置。 在那个超级重要的修补发布以后,你想要回到被打扰之前的工作。因为现在 hotfix 分支 和 master 指向相同的提交,现在没什么用了,可以先删掉它。使用 git branch 的 -d 选 项表示删除: $ git branch -d hotfix Deleted branch hotfix (3a0874c). 现在可以回到未完成的问题 #53 分支继续工作了(图3-15): 50 Scott Chacon Pro Git 3.2节 基本的分支与合并 $ git checkout iss53 Switched to branch "iss53" $ vim index.html $ git commit -a -m 'finished the new footer [issue 53]' [iss53]: created ad82d7a: "finished the new footer [issue 53]" 1 files changed, 1 insertions(+), 0 deletions(-) 图 3.15: iss53 分支可以不受影响继续推进。 不用担心 hotfix 分支的内容还没包含在 iss53 中。如果确实需要纳入此次修补,可以 用 git merge master 把 master 分支合并到 iss53,或者等完成后,再将 iss53 分支中 的更新并入 master。 3.2.2 基本合并 在问题 #53 相关的工作完成之后,可以合并回 master 分支,实际操作同前面合并 hotfix 分支差不多,只需检出想要更新的分支(master),并运行 git merge 命令指定来 源: $ git checkout master $ git merge iss53 Merge made by recursive. README | 1 + 1 files changed, 1 insertions(+), 0 deletions(-) 请注意,这次合并的实现,并不同于之前 hotfix 的并入方式。这一次,你的开发历史是 从更早的地方开始分叉的。由于当前 master 分支所指向的 commit (C4)并非想要并入分支 (iss53)的直接祖先,Git 不得不进行一些处理。就此例而言,Git 会用两个分支的末端 (C4 和 C5)和它们的共同祖先(C2)进行一次简单的三方合并计算。图 3.16 标出了 Git 在用于合并的三个更新快照: Git 没有简单地把分支指针右移,而是对三方合并的结果作一新的快照,并自动创建一个 指向它的 commit(C6)(见图 3.17)。我们把这个特殊的 commit 称作合并提交(merge commit),因为它的祖先不止一个。 值得一提的是 Git 可以自己裁决哪个共同祖先才是最佳合并基础;这和 CVS 或 Subver- sion(1.5 以后的版本)不同,它们需要开发者手工指定合并基础。所以此特性让 Git 的 合并操作比其他系统都要简单不少。 51 第3章 Git 分支 Scott Chacon Pro Git 图 3.16: Git 为分支合并自动识别出最佳的同源合并点。 图 3.17: Git 自动创建了一个包含了合并结果的 commit 对象。 既然你的工作成果已经合并了,iss53 也就没用了。你可以就此删除它,并在问题追踪系 统里把该问题关闭。 $ git branch -d iss53 3.2.3 冲突的合并 有时候合并操作并不会如此顺利。如果你修改了两个待合并分支里同一个文件的同一部 分,Git 就无法干净地把两者合到一起(译注:逻辑上说,这种问题只能由人来解决)。如 果你在解决问题 #53 的过程中修改了 hotfix 中修改的部分,将得到类似下面的结果: $ git merge iss53 Auto-merging index.html CONFLICT (content): Merge conflict in index.html Automatic merge failed; fix conflicts and then commit the result. Git 作了合并,但没有提交,它会停下来等你解决冲突。要看看哪些文件在合并时发生冲 突,可以用 git status 查阅: 52 Scott Chacon Pro Git 3.2节 基本的分支与合并 [master*]$ git status index.html: needs merge # On branch master # Changed but not updated: # (use "git add ..." to update what will be committed) # (use "git checkout -- ..." to discard changes in working directory) # # unmerged: index.html # 任何包含未解决冲突的文件都会以未合并(unmerged)状态列出。Git 会在有冲突的文件 里加入标准的冲突解决标记,可以通过它们来手工定位并解决这些冲突。可以看到此文件包 含类似下面这样的部分: <<<<<<< HEAD:index.html ======= >>>>>>> iss53:index.html 可以看到 ======= 隔开的上半部分,是 HEAD(即 master 分支,在运行 merge 命令时 检出的分支)中的内容,下半部分是在 iss53 分支中的内容。解决冲突的办法无非是二者 选其一或者由你亲自整合到一起。比如你可以通过把这段内容替换为下面这样来解决: 这个解决方案各采纳了两个分支中的一部分内容,而且我还删除了 <<<<<<<,=======, 和>>>>>>> 这些行。在解决了所有文件里的所有冲突后,运行 git add 将把它们标记为已 解决(resolved)。因为一旦暂存,就表示冲突已经解决。如果你想用一个有图形界面的工 具来解决这些问题,不妨运行 git mergetool,它会调用一个可视化的合并工具并引导你解 决所有冲突: $ git mergetool merge tool candidates: kdiff3 tkdiff xxdiff meld gvimdiff opendiff emerge vimdiff Merging the files: index.html Normal merge conflict for 'index.html': {local}: modified {remote}: modified Hit return to start merge resolution tool (opendiff): 53 第3章 Git 分支 Scott Chacon Pro Git 如果不想用默认的合并工具(Git 为我默认选择了 opendiff,因为我在 Mac 上运行了该 命令),你可以在上方“merge tool candidates(候选合并工具)”里找到可用的合并工 具列表,输入你想用的工具名。我们将在第七章讨论怎样改变环境中的默认值。 退出合并工具以后,Git 会询问你合并是否成功。如果回答是,它会为你把相关文件暂存 起来,以表明状态为已解决。 再运行一次 git status 来确认所有冲突都已解决: $ git status # On branch master # Changes to be committed: # (use "git reset HEAD ..." to unstage) # # modified: index.html # 如果觉得满意了,并且确认所有冲突都已解决,也就是进入了缓存区,就可以用 git commit 来完成这次合并提交。提交的记录差不多是这样: Merge branch 'iss53' Conflicts: index.html # # It looks like you may be committing a MERGE. # If this is not correct, please remove the file # .git/MERGE_HEAD # and try again. # 如果想给将来看这次合并的人一些方便,可以修改该信息,提供更多合并细节。比如你都 作了哪些改动,以及这么做的原因。有时候裁决冲突的理由并不直接或明显,有必要略加注 解。 3.3 分支管理 到目前为止,你已经学会了如何创建、合并和删除分支。除此之外,我们还需要学习如何 管理分支,在日后的常规工作中会经常用到下面介绍的管理命令。 git branch 命令不仅仅能创建和删除分支,如果不加任何参数,它会给出当前所有分支 的清单: $ git branch iss53 * master testing 54 Scott Chacon Pro Git 3.4节 分支式工作流程 注意看 master 分支前的 * 字符:它表示当前所在的分支。也就是说,如果现在提交 更新,master 分支将随着开发进度前移。若要查看各个分支最后一次 commit 信息,运行 git branch -v: $ git branch -v iss53 93b412c fix javascript issue * master 7a98805 Merge branch 'iss53' testing 782fd34 add scott to the author list in the readmes 要从该清单中筛选出你已经(或尚未)与当前分支合并的分支,可以用 --merge 和 -- no-merged 选项(Git 1.5.6 以上版本)。比如 git branch -merge 查看哪些分支已被并 入当前分支: $ git branch --merged iss53 * master 之前我们已经合并了 iss53,所以在这里会看到它。一般来说,列表中没有 * 的分支通 常都可以用 git branch -d 来删掉。原因很简单,既然已经把它们所包含的工作整合到了 其他分支,删掉也不会损失什么。 另外可以用 git branch --no-merged 查看尚未合并的工作: $ git branch --no-merged testing 我们会看到其余还未合并的分支。因为其中还包含未合并的工作,用 git branch -d 删 除该分支会导致失败: $ git branch -d testing error: The branch 'testing' is not an ancestor of your current HEAD. 不过,如果你坚信你要删除它,可以用大写的删除选项 -D 强制执行,例如 git branch -D testing。 3.4 分支式工作流程 如今有了分支与合并的基础,你可以(或应该)用它来做点什么呢?在本节,我们会介绍 些使用分支进行开发的工作流程。而正是由于分支管理的便捷,才衍生出了这类典型的工作 模式,有机会可以实践一下。 55 第3章 Git 分支 Scott Chacon Pro Git 3.4.1 长期分支 由于 Git 使用简单的三方合并,所以就算在较长一段时间内,反复多次把某个分支合并 到另一分支,也不是什么难事。也就是说,你可以同时拥有多个开放的分支,每个分支用于 完成特定的任务,随着开发的推进,你可以随时把某个特性分支的成果并到其他分支中。 许多使用 Git 的开发者都喜欢以这种方式来开展工作,比如仅在 master 分支中保留完 全稳定的代码,即已经发布或即将发布的代码。与此同时,他们还有一个名为 develop 或 next 的平行分支,专门用于后续的开发,或仅用于稳定性测试 —— 当然并不是说一定要 绝对稳定,不过一旦进入某种稳定状态,便可以把它合并到 master 里。这样,在确保这些 已完成的特性分支(短期分支,如前例的 iss53)能够通过所有测试,并且不会引入更多错 误之后,就可以并到主干分支中,等待下一次的发布。 本质上我们刚才谈论的,是随着 commit 不停前移的指针。稳定分支的指针总是在提交历 史中落后一大截,而前沿分支总是比较靠前(见图 3.18)。 图 3.18: 稳定分支总是比较老旧。 或者把它们想象成工作流水线可能会比较容易理解,经过测试的 commit 集合被遴选到更 稳定的流水线(见图 3.19)。 图 3.19: 想象成流水线可能会容易点。 你可以用这招维护不同层次的稳定性。某些大项目还会有个 proposed(建议)或 pu(proposed updates,建议更新)分支,它包含着那些可能还没有成熟到进入 next 或 master 的内容。这么做的目的是拥有不同层次的稳定性:当这些分支进入到更稳定的水 平时,再把它们合并到更高层分支中去。再次说明下,使用多个长期分支的做法并非必需, 不过一般来说,对于特大型项目或特复杂的项目,这么做确实更容易管理。 3.4.2 特性分支 在任何规模的项目中都可以使用特性(Topic)分支。一个特性分支是指一个短期的,用 来实现单一特性或与其相关工作的分支。可能你在以前的版本控制系统里从未做过类似这样 的事请,因为通常创建与合并分支消耗太大。然而在 Git 中,一天之内建立,使用,合并 再删除多个分支是常见的事。 我们在上节的例子里已经见过这种用法了。我们创建了 iss53 和 hotfix 这两个特性分 支,在提交了若干更新后,把它们合并到主干分支,然后删除。该技术允许你迅速且完全的 56 Scott Chacon Pro Git 3.5节 远程分支 进行语境切换 —— 因为你的工作分散在不同的流水线里,每个分支里的改变都和它的目标 特性相关,浏览代码之类的事情因而变得更简单了。你可以把作出的改变保持在特性分支中 几分钟,几天甚至几个月,等它们成熟以后再合并,而不用在乎它们建立的顺序或者进度。 现在我们来看一个实际的例子。请看图 3.20,起先我们在 master 工作到 C1,然后开始 一个新分支 iss91 尝试修复 91 号缺陷,提交到 C6 的时候,又冒出一个新的解决问题的 想法,于是从之前 C4 的地方又分出一个分支 iss91v2,干到 C8 的时候,又回到主干中 提交了 C9 和 C10,再回到 iss91v2 继续工作,提交 C11,接着,又冒出个不太确定的想 法,从 master 的最新提交 C10 处开了个新的分支 dumbidea 做些试验。 图 3.20: 拥有多个特性分支的提交历史。 现在,假定两件事情:我们最终决定使用第二个解决方案,即 iss91v2 中的办法;另 外,我们把 dumbidea 分支拿给同事们看了以后,发现它竟然是个天才之作。所以接下来, 我们抛弃原来的 iss91 分支(即丢弃 C5 和 C6),直接在主干中并入另外两个分支。最终 的提交历史将变成图 3.21 这样: 请务必牢记这些分支全部都是本地分支,这一点很重要。当你在使用分支及合并的时候, 一切都是在你自己的 Git 仓库中进行的 —— 完全不涉及与服务器的交互。 3.5 远程分支 远程分支(remote branch)是对远程仓库状态的索引。它们是一些无法移动的本地分 支;只有在进行 Git 的网络活动时才会更新。远程分支就像是书签,提醒着你上次连接远 程仓库时上面各分支的位置。 我们用 (远程仓库名)/(分支名) 这样的形式表示远程分支。比如我们想看看上次同 origin 仓库通讯时 master 的样子,就应该查看 origin/master 分支。如果你和同伴一起 修复某个问题,但他们先推送了一个 iss53 分支到远程仓库,虽然你可能也有一个本地的 iss53 分支,但指向服务器上最新更新的却应该是 origin/iss53 分支。 可能有点乱,我们不妨举例说明。假设你们团队有个地址为 git.ourcompany.com 的 Git 服务器。如果你从这里克隆,Git 会自动为你将此远程仓库命名为 origin,并下载其中所 有的数据,建立一个指向它的 master 分支的指针,在本地命名为 origin/master,但你无 法在本地更改其数据。接着,Git 建立一个属于你自己的本地 master 分支,始于 origin 57 第3章 Git 分支 Scott Chacon Pro Git 图 3.21: 合并了 dumbidea 和 iss91v2 以后的历史。 上 master 分支相同的位置,你可以就此开始工作(见图 3.22): 图 3.22: 一次 Git 克隆会建立你自己的本地分支 master 和远程分支 origin/master,它们都指向 origin/master 分支的最后一次提交。 要是你在本地 master 分支做了会儿事情,与此同时,其他人向 git.ourcompany.com 推 送了内容,更新了上面的 master 分支,那么你的提交历史会开始朝不同的方向发展。不过 只要你不和服务器通讯,你的 origin/master 指针不会移动(见图 3.23)。 可以运行 git fetch origin 来进行同步。该命令首先找到 origin 是哪个服务器(本例 为 git.ourcompany.com),从上面获取你尚未拥有的数据,更新你本地的数据库,然后把 origin/master 的指针移到它最新的位置(见图 3.24)。 为了演示拥有多个远程分支(不同的远程服务器)的项目是个什么样,我们假设你还有 58 Scott Chacon Pro Git 3.5节 远程分支 图 3.23: 在本地工作的同时有人向远程仓库推送内容会让提交历史发生分歧。 图 3.24: git fetch 命令会更新 remote 索引。 另一个仅供你的敏捷开发小组使用的内部服务器 git.team1.ourcompany.com。可以用第二 章中提到的 git remote add 命令把它加为当前项目的远程分支之一。我们把它命名为 teamone,表示那一整串 Git 地址(见图 3.25)。 现在你可以用 git fetch teamone 来获取小组服务器上你还没有的数据了。由于当前 该服务器上的内容是你 origin 服务器上的子集,Git 不会下载任何数据,而只是简单地 创建一个名为 teamone/master 的分支来指向 teamone 服务器上 master 所指向的更新 31b8e(见图 3.26)。 3.5.1 推送 要想和其他人分享某个分支,你需要把它推送到一个你拥有写权限的远程仓库。你的本地 分支不会被自动同步到你引入的远程分支中,除非你明确执行推送操作。换句话说,对于无 意分享的,你尽可以保留为私人分支,而只推送那些协同工作的特性分支。 59 第3章 Git 分支 Scott Chacon Pro Git 图 3.25: 把另一个服务器加为远程仓库 图 3.26: 你在本地有了一个指向 teamone 服务器上 master 分支的索引。 如果你有个叫 serverfix 的分支需要和他人一起开发,可以运行 git push (远程仓库 名) (分支名): $ git push origin serverfix Counting objects: 20, done. Compressing objects: 100% (14/14), done. Writing objects: 100% (15/15), 1.74 KiB, done. Total 15 (delta 5), reused 0 (delta 0) To git@github.com:schacon/simplegit.git * [new branch] serverfix -> serverfix 这其实有点像条捷径。Git 自动把 serverfix 分支名扩展为 refs/heads/server- fix:refs/heads/serverfix,意为“取出我的 serverfix 本地分支,推送它来更新远程 仓库的 serverfix 分支”。我们将在第九章进一步介绍 refs/heads/ 部分的细节,不过 一般使用的时候都可以省略它。也可以运行 git push origin serverfix:serferfix 来 60 Scott Chacon Pro Git 3.5节 远程分支 实现相同的效果,它的意思是“提取我的 serverfix 并更新到远程仓库的 serverfix”。 通过此语法,你可以把本地分支推送到某个命名不同的远程分支:若想把远程分支叫作 awesomebranch,可以用 git push origin serverfix:awesomebranch 来推送数据。 接下来,当你的协作者再次从服务器上获取数据时,他们将得到一个新的远程分支 origin/serverfix: $ git fetch origin remote: Counting objects: 20, done. remote: Compressing objects: 100% (14/14), done. remote: Total 15 (delta 5), reused 0 (delta 0) Unpacking objects: 100% (15/15), done. From git@github.com:schacon/simplegit * [new branch] serverfix -> origin/serverfix 值得注意的是,在 fetch 操作抓来新的远程分支之后,你仍然无法在本地编辑该远程仓 库。换句话说,在本例中,你不会有一个新的 serverfix 分支,有的只是一个你无法移动 的 origin/serverfix 指针。 如果要把该内容合并到当前分支,可以运行 git merge origin/serverfix。如果想要一 份自己的 serverfix 来开发,可以在远程分支的基础上分化出一个新的分支来: $ git checkout -b serverfix origin/serverfix Branch serverfix set up to track remote branch refs/remotes/origin/serverfix. Switched to a new branch "serverfix" 这会切换到新建的 serverfix 本地分支,其内容同远程分支 origin/serverfix 一致, 你可以在里面继续开发了。 3.5.2 跟踪分支 从远程分支检出的本地分支,称为跟踪分支(tracking branch)。跟踪分支是一种和远程 分支有直接联系的本地分支。在跟踪分支里输入 git push,Git 会自行推断应该向哪个服 务器的哪个分支推送数据。反过来,在这些分支里运行 git pull 会获取所有远程索引,并 把它们的数据都合并到本地分支中来。 在克隆仓库时,Git 通常会自动创建一个 master 分支来跟踪 origin/master。这正是 git push 和 git pull 一开始就能正常工作的原因。当然,你可以随心所欲地设定为其它 跟踪分支,比如 origin 上除了 master 之外的其它分支。刚才我们已经看到了这样的一个 例子:git checkout -b [分支名] [远程名]/[分支名]。如果你有 1.6.2 以上版本的 Git, 还可以用 --track 选项简化: $ git checkout --track origin/serverfix Branch serverfix set up to track remote branch refs/remotes/origin/serverfix. Switched to a new branch "serverfix" 要为本地分支设定不同于远程分支的名字,只需在前个版本的命令里换个名字: 61 第3章 Git 分支 Scott Chacon Pro Git $ git checkout -b sf origin/serverfix Branch sf set up to track remote branch refs/remotes/origin/serverfix. Switched to a new branch "sf" 现在你的本地分支 sf 会自动向 origin/serverfix 推送和抓取数据了。 3.5.3 删除远程分支 如果不再需要某个远程分支了,比如搞定了某个特性并把它合并进了远程的 master 分 支(或任何其他存放稳定代码的地方),可以用这个非常无厘头的语法来删除它:git push [远程名] :[分支名]。如果想在服务器上删除 serverfix 分支,运行下面的命令: $ git push origin :serverfix To git@github.com:schacon/simplegit.git - [deleted] serverfix 咚!服务器上的分支没了。你最好特别留心这一页,因为你一定会用到那个命令,而且你 很可能会忘掉它的语法。有种方便记忆这条命令的方法:记住我们不久前见过的 git push [远程名] [本地分支]:[远程分支] 语法,如果省略 [本地分支],那就等于是在说“在这里 提取空白然后把它变成[远程分支]”。 3.6 衍合 把一个分支整合到另一个分支的办法有两种:merge(合并) 和 rebase(衍合)。在本 章我们会学习什么是衍合,如何使用衍合,为什么衍合操作如此富有魅力,以及我们应该在 什么情况下使用衍合。 3.6.1 衍合基础 请回顾之前有关合并的一节(见图 3.27),你会看到开发进程分叉到两个不同分支,又 各自提交了更新。 图 3.27: 最初分叉的提交历史。 之前介绍过,最容易的整合分支的方法是 merge 命令,它会把两个分支最新的快照(C3 和 C4)以及二者最新的共同祖先(C2)进行三方合并。如图 3.28 所示: 62 Scott Chacon Pro Git 3.6节 衍合 图 3.28: 通过合并一个分支来整合分叉了的历史。 其实,还有另外一个选择:你可以把在 C3 里产生的变化补丁重新在 C4 的基础上打一 遍。在 Git 里,这种操作叫做衍合(rebase)。有了 rebase 命令,就可以把在一个分支 里提交的改变在另一个分支里重放一遍。 在这个例子里,可以运行下面的命令: $ git checkout experiment $ git rebase master First, rewinding head to replay your work on top of it... Applying: added staged command 它的原理是回到两个分支(你所在的分支和你想要衍合进去的分支)的共同祖先,提取你 所在分支每次提交时产生的差异(diff),把这些差异分别保存到临时文件里,然后从当前 分支转换到你需要衍合入的分支,依序施用每一个差异补丁文件。图 3.29 演示了这一过 程: 图 3.29: 把 C3 里产生的改变衍合到 C4 中。 现在,你可以回到 master 分支然后进行一次快进合并(见图 3.30): 图 3.30: master 分支的快进。 现在,合并后的 C3(即现在的 C3’)所指的快照,同三方合并例子中的 C5 所指的快照 内容一模一样了。最后整合得到的结果没有任何区别,但衍合能产生一个更为整洁的提交历 63 第3章 Git 分支 Scott Chacon Pro Git 史。如果视察一个衍合过的分支的历史记录,看起来更清楚:仿佛所有修改都是先后进行 的,尽管实际上它们原来是同时发生的。 你可以经常使用衍合,确保在远程分支里的提交历史更清晰。比方说,某些项目自己不是 维护者,但想帮点忙,就应该尽可能使用衍合:先在一个分支里进行开发,当准备向主项目 提交补丁的时候,再把它衍合到 origin/master 里面。这样,维护者就不需要做任何整合 工作,只需根据你提供的仓库地址作一次快进,或者采纳你提交的补丁。 请注意,合并结果中最后一次提交所指向的快照,无论是通过一次衍合还是一次三方合 并,都是同样的快照内容,只是提交的历史不同罢了。衍合按照每行改变发生的次序重演发 生的改变,而合并是把最终结果合在一起。 3.6.2 更多有趣的衍合 你还可以在衍合分支以外的地方衍合。以图 3.31 的历史为例。你创建了一个特性分支 server 来给服务器端代码添加一些功能,然后提交 C3 和 C4。然后从 C3 的地方再增加一 个 client 分支来对客户端代码进行一些修改,提交 C8 和 C9。最后,又回到 server 分 支提交了 C10。 图 3.31: 从一个特性分支里再分出一个特性分支的历史。 假设在接下来的一次软件发布中,你决定把客户端的修改先合并到主线中,而暂缓并入服 务端软件的修改(因为还需要进一步测试)。你可以仅提取对客户端的改变(C8 和 C9), 然后通过使用 git rebase 的 --onto 选项来把它们在 master 分支上重演: $ git rebase --onto master server client 这基本上等于在说“检出 client 分支,找出 client 分支和 server 分支的共同祖先之 后的变化,然后把它们在 master 上重演一遍”。是不是有点复杂?不过它的结果,如图 3.32 所示,非常酷: 现在可以快进 master 分支了(见图 3.33): 64 Scott Chacon Pro Git 3.6节 衍合 图 3.32: 衍合一个特性分支上的另一个特性分支。 $ git checkout master $ git merge client 图 3.33: 快进 master 分支,使之包含 client 分支的变化。 现在你决定把 server 分支的变化也包含进来。可以直接把 server 分支衍合到 master 而不用手工转到 server 分支再衍合。git rebase [主分支] [特性分支] 命令会先检出特 性分支 server,然后在主分支 master 上重演: $ git rebase master server 于是 server 的进度应用到 master 的基础上,如图 3.34: 图 3.34: 在 master 分支上衍合 server 分支。 然后快进主分支 master: 65 第3章 Git 分支 Scott Chacon Pro Git $ git checkout master $ git merge server 现在 client 和 server 分支的变化都被整合了,不妨删掉它们,把你的提交历史变成图 3.35 的样子: $ git branch -d client $ git branch -d server 图 3.35: 最终的提交历史 3.6.3 衍合的风险 呃,奇妙的衍合也不是完美无缺的,一句话可以总结这点: 永远不要衍合那些已经推送到公共仓库的更新。 如果你遵循这条金科玉律,就不会出差错。否则,人民群众会仇恨你,你的朋友和家人也 会嘲笑你,唾弃你。 在衍合的时候,实际上抛弃了一些现存的 commit 而创造了一些类似但不同的新 commit。 如果你把commit 推送到某处然后其他人下载并在其基础上工作,然后你用 git rebase 重 写了这些commit 再推送一次,你的合作者就不得不重新合并他们的工作,这样当你再次从 他们那里获取内容的时候事情就会变得一团糟。 下面我们用一个实际例子来说明为什么公开的衍合会带来问题。假设你从一个中央服务器 克隆然后在它的基础上搞了一些开发,提交历史类似图 3.36: 图 3.36: 克隆一个仓库,在其基础上工作一番。 66 Scott Chacon Pro Git 3.6节 衍合 现在,其他人进行了一些包含一次合并的工作(得到结果 C6),然后把它推送到了中央 服务器。你获取了这些数据并把它们合并到你本地的开发进程里,让你的历史变成类似图 3.37 这样: 图 3.37: 获取更多提交,并入你的开发进程。 接下来,那个推送 C6 上来的人决定用衍合取代那次合并;他们用 git push --force 覆 盖了服务器上的历史,得到 C4’。然后你再从服务器上获取更新: 图 3.38: 有人推送了衍合过的 C4’,丢弃了你作为开发基础的 C6。 这时候,你需要再次合并这些内容,尽管之前已经做过一次了。衍合会改变这些 commit 的 SHA-1 校验值,这样 Git 会把它们当作新的 commit,然而这时候在你的提交历史早就 有了 C4 的内容(见图 3.39): 你迟早都是要并入其他协作者提交的内容的,这样才能保持同步。当你做完这些,你的提 交历史里会同时包含 C4 和 C4’,两者有着不同的 SHA-1 校验值,但却拥有一样的作者日 期与提交说明,令人费解!更糟糕的是,当你把这样的历史推送到服务器,会再次把这些衍 合的提交引入到中央服务器,进一步迷惑其他人。 如果把衍合当成一种在推送之前清理提交历史的手段,而且仅仅衍合那些永远不会公开的 commit,那就不会有任何问题。如果衍合那些已经公开的 commit,而与此同时其他人已经 用这些 commit 进行了后续的开发工作,那你有得麻烦了。 67 第3章 Git 分支 Scott Chacon Pro Git 图 3.39: 你把相同的内容又合并了一遍,生成一个新的提交 C8。 3.7 小结 读到这里,你应该已经学会了如何创建分支并切换到新分支;在不同分支间转换;合并本 地分支;把分支推送到共享服务器上,同世界分享;使用共享分支与他人协作;以及在分享 之前进行衍合。 68 第4章 服务器上的 Git 到目前为止,你应该已经学会了使用 Git 来完成日常的工作。然而,如果想与他人合 作,还需要一个远程的 Git 仓库。尽管技术上可以从个人的仓库里推送和拉取改变,但是 我们不鼓励这样做,因为一不留心就很容易弄混其他人的进度。另外,你也一定希望合作者 们即使在自己不开机的时候也能从仓库获取数据——拥有一个更稳定的公共仓库十分有用。 因此,更好的合作方式是建立一个大家都可以访问的共享仓库,从那里推送和拉取数据。我 们将把这个仓库称为 “Git 服务器”;代理一个 Git 仓库只需要花费很少的资源,几乎从 不需要整个服务器来支持它的运行。 架设一个 Git 服务器不难。第一步是选择与服务器通讯的协议。本章的第一节将介绍可 用的协议以及他们各自的优缺点。下面一节将介绍一些针对各个协议典型的设置以及如何在 服务器上运行它们。最后,如果你不介意在其他人的服务器上保存你的代码,又不想经历自 己架设和维护服务器的麻烦,我们将介绍几个网络上的仓库托管服务。 如果你对架设自己的服务器没兴趣,可以跳到本章最后一节去看看如何创建一个代码托管 账户然后继续下一章,我们会在那里讨论一个分布式源码控制环境的林林总总。 远程仓库通常只是一个 纯仓库(bare repository) ——一个没有当前工作目录的仓库。 因为该仓库只是一个合作媒介,所以不需要从一个处于已从硬盘上检出状态的快照;仓库里 仅仅是 Git 的数据。简单的说,纯仓库是你项目里 .git 目录的内容,别无他物。 4.1 协议 Git 可以使用四种主要的协议来传输数据:本地传输,SSH 协议,Git 协议和 HTTP 协 议。下面分别介绍一下他们以及你应该(或不应该)在怎样的情形下使用他们。 值得注意的是除了 HTTP 协议之外,其他所有协议都要求在服务器端安装并运行 Git 。 4.1.1 本地协议 最基础的就是 本地协议(Local protocol) 了,远程仓库在该协议中就是硬盘上的另一个 目录。这常见于团队每一个成员都对一个共享的文件系统(例如 NFS )拥有访问权,抑或比 较少见的多人共用同一台电脑的时候。后者不是很理想,因为你所有的代码仓库实例都储存 在同一台电脑里,增加了灾难性数据损失的可能性。 如果你使用一个共享的文件系统,就可以在一个本地仓库里克隆,推送和获取。要从这样 的仓库里克隆或者将其作为远程仓库添加现有工程里,可以用指向该仓库的路径作为URL。 比如,克隆一个本地仓库,可以用如下命令完成: 69 第4章 服务器上的 Git Scott Chacon Pro Git $ git clone /opt/git/project.git 或者这样: $ git clone file:///opt/git/project.git 如果你在URL的开头明确的使用 file:// ,那么 Git 会以一种略微不同的方式运行。 如果你只给出路径,Git 会尝试使用硬链接或者直接复制它需要的文件。如果使用了 file:// ,Git会调用它平时通过网络来传输数据的工序,而这种方式的效率相对很低。使 用 file:// 前缀的主要原因是当你需要一个不包含无关引用或对象的干净仓库副本的时候 ——一般是从其他版本控制系统的导入之后或者类似的情形(参见第9章的维护任务)。我 们这里使用普通路径,因为通常这样总是更快。 要添加一个本地仓库到现有 Git 工程,运行如下命令: $ git remote add local_proj /opt/git/project.git 然后就可以像在网络上一样向这个远程仓库推送和获取数据了。 优点 基于文件仓库的优点在于它的简单,同时保留了现存文件的权限和网络访问权限。如果你 的团队已经有一个全体共享的文件系统,建立仓库就十分容易了。你只需把一份纯仓库的副 本放在大家能访问的地方,然后像对其他共享目录一样设置读写权限就可以了。我们将在下 一节“在服务器上部署 Git”中讨论如何为此导出一个纯仓库的副本。 这也是个从别人工作目录里获取他工作成果的快捷方法。假如你和你的同事在一个项目中 合作,他们想让你检出一些东西的时候,运行类似 git pull /home/john/project 通常会 比他们推送到服务器,而你又从服务器获取简单得多。 缺点 这种方法的缺点是,与基本的网络连接访问相比,能从不同的位置访问的共享权限难以架 设。如果你想从家里的笔记本电脑上推送,就要先挂载远程硬盘,这和基于网络连接的访问 相比更加困难和缓慢。 另一个很重要的问题是该方法不一定就是最快的,尤其是对于共享挂载的文件系统。本地 仓库只有在你对数据访问速度快的时候才快。在同一个服务器上,如果二者同时允许 Git 访问本地硬盘,通过 NFS 访问仓库通常会比 SSH 慢。 4.1.2 SSH 协议 Git 使用的传输协议中最常见的可能就是 SSH 了。这是因为大多数环境已经支持通过 SSH 对服务器的访问——即使还没有,也很容易架设。SSH 也是唯一一个同时便于读和写操 作的网络协议。另外两个网络协议(HTTP 和 Git)通常都是只读的,所以虽然二者对大多 70 Scott Chacon Pro Git 4.1节 协议 数人都可用,但执行写操作时还是需要 SSH。SSH 同时也是一个验证授权的网络协议;而因 为其普遍性,通常也很容易架设和使用。 通过 SSH 克隆一个 Git 仓库,你可以像下面这样给出 ssh:// 的 URL: $ git clone ssh://user@server:project.git 或者不指明某个协议——这时 Git 会默认使用 SSH : $ git clone user@server:project.git 也可以不指明用户,Git 会默认使用你当前登录的用户。 优点 使用 SSH 的好处有很多。首先,如果你想拥有对网络仓库的写权限,基本上不可能不使 用 SSH。其次,SSH 架设相对比较简单—— SSH 守护进程很常见,很多网络管理员都有一 些使用经验,而且很多操作系统都自带了它或者相关的管理工具。再次,通过 SSH 进行访 问是安全的——所有数据传输都是加密和授权的。最后,类似 Git 和 本地协议,SSH 很高 效,会在传输之前尽可能的压缩数据。 缺点 SSH 的限制在于你不能通过它实现仓库的匿名访问。即使仅为读取数据,人们也必须在能 通过 SSH 访问主机的前提下才能访问仓库,这使得 SSH 不利于开源的项目。如果你仅仅在 公司网络里使用,SSH 可能是你唯一需要使用的协议。如果想允许对项目的匿名只读访问, 那么除了为自己推送而架设 SSH 协议之外,还需要其他协议来让别人获取数据。 4.1.3 Git 协议 接下来是 Git 协议。这是一个包含在 Git 软件包中的特殊守护进程; 它会监听一个提 供类似于 SSH 服务的特定端口(9418),而无需任何授权。用 Git 协议运营仓库,你需要 创建 git-export-daemon-ok 文件——它是协议进程提供仓库服务的必要条件——但除此之 外该服务没有什么安全措施。要么所有人都能克隆 Git 仓库,要么谁也不能。这也意味着 该协议通常不能用来进行推送。你可以允许推送操作;然而由于没有授权机制,一旦允许 该操作,网络上任何一个知道项目 URL 的人将都有推送权限。不用说,这是十分罕见的情 况。 优点 Git 协议是现存最快的传输协议。如果你在提供一个有很大访问量的公共项目,或者一 个不需要对读操作进行授权的庞大项目,架设一个 Git 守护进程来供应仓库是个不错的选 择。它使用与 SSH 协议相同的数据传输机制,但省去了加密和授权的开销。 71 第4章 服务器上的 Git Scott Chacon Pro Git 缺点 Git 协议消极的一面是缺少授权机制。用 Git 协议作为访问项目的唯一方法通常是不可 取的。一般做法是,同时提供 SSH 接口,让几个开发者拥有推送(写)权限,其他人通过 git:// 拥有只读权限。 Git 协议可能也是最难架设的协议。它要求有单独的守护进程,需 要定制——我们将在本章的 “Gitosis” 一节详细介绍它的架设——需要设定 xinetd 或 类似的程序,而这些就没那么平易近人了。该协议还要求防火墙开放 9418 端口,而企业级 防火墙一般不允许对这个非标准端口的访问。大型企业级防火墙通常会封锁这个少见的端 口。 4.1.4 HTTP/S 协议 最后还有 HTTP 协议。HTTP 或 HTTPS 协议的优美之处在于架设的简便性。基本上, 只 需要把 Git 的纯仓库文件放在 HTTP 的文件根目录下,配置一个特定的 post-update 挂钩 (hook),就搞定了(Git 挂钩的细节见第七章)。从此,每个能访问 Git 仓库所在服务 器上的 web 服务的人都可以进行克隆操作。下面的操作可以允许通过 HTTP 对仓库进行读 取: $ cd /var/www/htdocs/ $ git clone --bare /path/to/git_project gitproject.git $ cd gitproject.git $ mv hooks/post-update.sample hooks/post-update $ chmod a+x hooks/post-update 这样就可以了。Git 附带的 post-update 挂钩会默认运行合适的命令(git update- server-info)来确保通过 HTTP 的获取和克隆正常工作。这条命令在你用 SSH 向仓库推送 内容时运行;之后,其他人就可以用下面的命令来克隆仓库: $ git clone http://example.com/gitproject.git 在本例中,我们使用了 Apache 设定中常用的 /var/www/htdocs 路径,不过你可以使用 任何静态 web 服务——把纯仓库放在它的目录里就行了。 Git 的数据是以最基本的静态文 件的形式提供的(关于如何提供文件的详情见第9章)。 通过HTTP进行推送操作也是可能的,不过这种做法不太常见并且牵扯到复杂的 WebDAV 设定。由于很少用到,本书将略过对该内容的讨论。如果对 HTTP 推送协议感兴趣,不妨 在这个地址看一下操作方法:http://www.kernel.org/pub/software/scm/git/docs/howto/ setup-git-server-over-http.txt 。通过 HTTP 推送的好处之一是你可以使用任何 WebDAV 服务器,不需要为 Git 设定特殊环境;所以如果主机提供商支持通过 WebDAV 更新网站内 容,你也可以使用这项功能。 优点 使用 HTTP 协议的好处是易于架设。几条必要的命令就可以让全世界读取到仓库的内容。 花费不过几分钟。HTTP 协议不会占用过多服务器资源。因为它一般只用到静态的 HTTP 服 务提供所有的数据,普通的 Apache 服务器平均每秒能供应数千个文件——哪怕是让一个小 型的服务器超载都很难。 72 Scott Chacon Pro Git 4.2节 在服务器部署 Git 你也可以通过 HTTPS 提供只读的仓库,这意味着你可以加密传输内容;你甚至可以要求 客户端使用特定签名的 SSL 证书。一般情况下,如果到了这一步,使用 SSH 公共密钥可能 是更简单的方案;不过也存在一些特殊情况,这时通过 HTTPS 使用带签名的 SSL 证书或者 其他基于 HTTP 的只读连接授权方式是更好的解决方案。 HTTP 还有个额外的好处:HTTP 是一个如此常见的协议,以至于企业级防火墙通常都允许 其端口的通信。 缺点 HTTP 协议的消极面在于,相对来说客户端效率更低。克隆或者下载仓库内容可能会花费 更多时间,而且 HTTP 传输的体积和网络开销比其他任何一个协议都大。因为它没有按需 供应的能力——传输过程中没有服务端的动态计算——因而 HTTP 协议经常会被称为 傻瓜 (dumb) 协议。更多 HTTP 协议和其他协议效率上的差异见第九章。 4.2 在服务器部署 Git 开始架设 Git 服务器的时候,需要把一个现存的仓库导出为新的纯仓库——不包含当前 工作目录的仓库。方法非常直截了当。 把一个仓库克隆为纯仓库,可以使用 clone 命令的 --bare 选项。纯仓库的目录名以 .git 结尾, 如下: $ git clone --bare my_project my_project.git Initialized empty Git repository in /opt/projects/my_project.git/ 该命令的输出有点迷惑人。由于 clone 基本上等于 git init 加 git fetch,这里出现 的就是 git init 的输出,它建立了一个空目录。实际的对象转换不会有任何输出,不过确 实发生了。现在在 my_project.git 中已经有了一份 Git 目录数据的副本。 大体上相当于 $ cp -Rf my_project/.git my_project.git 在配置文件中有几个小改变;不过从效果角度讲,克隆的内容是一样的。它仅包含了 Git 目录,没有工作目录,并且专门为之(译注: Git 目录)建立了一个单独的目录。 4.2.1 将纯目录转移到服务器 有了仓库的纯副本以后,剩下的就是把它放在服务器上并设定相关的协议。假设一个域 名为 git.example.com 的服务器已经架设好,并可以通过 SSH 访问,而你想把所有的 Git 仓库储存在 /opt/git 目录下。只要把纯仓库复制上去: $ scp -r my_project.git user@git.example.com:/opt/git 现在,其他对该服务器具有 SSH 访问权限并可以读取 /opt/git 的用户可以用以下命令 克隆: 73 第4章 服务器上的 Git Scott Chacon Pro Git $ git clone user@git.example.com:/opt/git/my_project.git 假如一个 SSH 用户对 /opt/git/my_project.git 目录有写权限,他会自动具有推送权 限。这时如果运行 git init 命令的时候加上 --shared 选项,Git 会自动对该仓库加入可 写的组。 $ ssh user@git.example.com $ cd /opt/git/my_project.git $ git init --bare --shared 可见选择一个 Git 仓库,创建一个纯的版本,最后把它放在你和同事都有 SSH 访问权的 服务器上是多么容易。现在已经可以开始在同一项目上密切合作了。 值得注意的是,这的的确确是架设一个少数人具有连接权的 Git 服务的全部——只要在 服务器上加入可以用 SSH 接入的帐号,然后把纯仓库放在大家都有读写权限的地方。一切 都做好了,无须更多。 下面的几节中,你会了解如何扩展到更复杂的设定。这些内容包含如何避免为每一个用 户建立一个账户,给仓库添加公共读取权限,架设网页界面,使用 Gitosis 工具等等。然 而,只是和几个人在一个不公开的项目上合作的话,仅仅是一个 SSH 服务器和纯仓库就足 够了,请牢记这一点。 4.2.2 小型安装 如果设备较少或者你只想在小型的开发团队里尝试 Git ,那么一切都很简单。架设 Git 服务最复杂的方面之一在于账户管理。如果需要仓库对特定的用户可读,而给另一部分用户 读写权限,那么访问和许可的安排就比较困难。 SSH 连接 如果已经有了一个所有开发成员都可以用 SSH 访问的服务器,架设第一个服务器将变得 异常简单,几乎什么都不用做(正如上节中介绍的那样)。如果需要对仓库进行更复杂的访 问控制,只要使用服务器操作系统的本地文件访问许可机制就行了。 如果需要团队里的每个人都对仓库有写权限,又不能给每个人在服务器上建立账户,那么 提供 SSH 连接就是唯一的选择了。我们假设用来共享仓库的服务器已经安装了 SSH 服务, 而且你通过它访问服务器。 有好几个办法可以让团队的每个人都有访问权。第一个办法是给每个人建立一个账户,直 截了当但过于繁琐。反复的运行 adduser 并且给所有人设定临时密码可不是好玩的。 第二个办法是在主机上建立一个 git 账户,让每个需要写权限的人发送一个 SSH 公钥, 然后将其加入 git 账户的 ~/.ssh/authorized_keys 文件。这样一来,所有人都将通过 git 账户访问主机。这丝毫不会影响提交的数据——访问主机用的身份不会影响commit的记录。 另一个办法是让 SSH 服务器通过某个 LDAP 服务,或者其他已经设定好的集中授权机 制,来进行授权。只要每个人都能获得主机的 shell 访问权,任何可用的 SSH 授权机制都 能达到相同效果。 74 Scott Chacon Pro Git 4.3节 生成 SSH 公钥 4.3 生成 SSH 公钥 话虽如此,大多数 Git 服务器使用 SSH 公钥来授权。为了得到授权,系统中的每个没有 公钥用户都得生成一个新的。该过程在所有操作系统上都差不多。 首先,确定一下是否已 经有一个公钥了。SSH 公钥默认储存在账户的 ~/.ssh 目录。进入那里并查看其内容,有没 有公钥一目了然: $ cd ~/.ssh $ ls authorized_keys2 id_dsa known_hosts config id_dsa.pub 关键是看有没有用 文件名 和 文件名.pub 来命名的一对文件,这个 文件名 通常是 id_dsa 或者 id_rsa。.pub 文件是公钥,另一个文件是密钥。假如没有这些文件(或者干 脆连 .ssh 目录都没有),你可以用 ssh-keygen 的程序来建立它们,该程序在 Linux/Mac 系统由 SSH 包提供, 在 Windows 上则包含在 MSysGit 包里: $ ssh-keygen Generating public/private rsa key pair. Enter file in which to save the key (/Users/schacon/.ssh/id_rsa): Enter passphrase (empty for no passphrase): Enter same passphrase again: Your identification has been saved in /Users/schacon/.ssh/id_rsa. Your public key has been saved in /Users/schacon/.ssh/id_rsa.pub. The key fingerprint is: 43:c5:5b:5f:b1:f1:50:43:ad:20:a6:92:6a:1f:9a:3a schacon@agadorlaptop.local 它先要求你确认保存公钥的位置(.ssh/id_rsa),然后它会让你重复一个密码两次,如 果不想在使用公钥的时候输入密码,可以留空。 现在,所有做过这一步的用户都得把它们的公钥给你或者 Git 服务器的管理者(假设 SSH 服务被设定为使用公钥机制)。他们只需要复制 .put 文件的内容然后 e-email 之。 公钥的样子大致如下: $ cat ~/.ssh/id_rsa.pub ssh-rsa AAAAB3NzaC1yc2EAAAABIwAAAQEAklOUpkDHrfHY17SbrmTIpNLTGK9Tjom/BWDSU GPl+nafzlHDTYW7hdI4yZ5ew18JH4JW9jbhUFrviQzM7xlELEVf4h9lFX5QVkbPppSwg0cda3 Pbv7kOdJ/MTyBlWXFCR+HAo3FXRitBqxiX1nKhXpHAZsMciLq8V6RjsNAQwdsdMFvSlVK/7XA t3FaoJoAsncM1Q9x5+3V0Ww68/eIFmb1zuUFljQJKprrX88XypNDvjYNby6vw/Pb0rwert/En mZ+AW4OZPnTPI89ZPmVMLuayrD2cE86Z/il8b+gw3r3+1nKatmIkjn2so1d01QraTlMqVSsbx NrRFi9wrf+M7Q== schacon@agadorlaptop.local 关于在多个操作系统上设立相同 SSH 公钥的教程,可以在 GitHub 有关 SSH 公钥的向导 中找到:http://github.com/guides/providing-your-ssh-key。 75 第4章 服务器上的 Git Scott Chacon Pro Git 4.4 架设服务器 现在我们过一边服务器端架设 SSH 访问的流程。本例将使用 authorized_keys 方法来 给用户授权。我们还将假定使用类似 Ubuntu 这样的标准 Linux 发行版。首先,创建一个 ‘git’ 用户并为其创建一个 .ssh 目录(译注:在用户的主目录下)。 $ sudo adduser git $ su git $ cd $ mkdir .ssh 接下来,把开发者的 SSH 公钥添加到这个用户的 authorized_keys 文件中。假设你通过 e-mail 收到了几个公钥并存到了临时文件里。重复一下,公钥大致看起来是这个样子: $ cat /tmp/id_rsa.john.pub ssh-rsa AAAAB3NzaC1yc2EAAAADAQABAAABAQCB007n/ww+ouN4gSLKssMxXnBOvf9LGt4L ojG6rs6hPB09j9R/T17/x4lhJA0F3FR1rP6kYBRsWj2aThGw6HXLm9/5zytK6Ztg3RPKK+4k Yjh6541NYsnEAZuXz0jTTyAUfrtU3Z5E003C4oxOj6H0rfIF1kKI9MAQLMdpGW1GYEIgS9Ez Sdfd8AcCIicTDWbqLAcU4UpkaX8KyGlLwsNuuGztobF8m72ALC/nLF6JLtPofwFBlgc+myiv O7TCUSBdLQlgMVOFq1I2uPWQOkOWQAHukEOmfjy2jctxSDBQ220ymjaNsHT4kgtZg2AYYgPq dAv8JggJICUvax2T9va5 gsg-keypair 只要把它们加入 authorized_keys 文件(译注:本例加入到了文件尾部): $ cat /tmp/id_rsa.john.pub >> ~/.ssh/authorized_keys $ cat /tmp/id_rsa.josie.pub >> ~/.ssh/authorized_keys $ cat /tmp/id_rsa.jessica.pub >> ~/.ssh/authorized_keys 现在可以使用 --bare 选项运行 git init 来设定一个空仓库,这会初始化一个不包含工 作目录的仓库。 $ cd /opt/git $ mkdir project.git $ cd project.git $ git --bare init 这时,Join,Josie 或者 Jessica 就可以把它加为远程仓库,推送一个分支,从而把第 一个版本的工程上传到仓库里了。值得注意的是,每次添加一个新项目都需要通过 shell 登入主机并创建一个纯仓库。我们不妨以 gitserver 作为 git 用户和仓库所在的主机名。 如果你在网络内部运行该主机,并且在 DNS 中设定 gitserver 指向该主机,那么以下这些 命令都是可用的: 76 Scott Chacon Pro Git 4.4节 架设服务器 # 在 John 的电脑上 $ cd myproject $ git init $ git add . $ git commit -m 'initial commit' $ git remote add origin git@gitserver:/opt/git/project.git $ git push origin master 这样,其他人的克隆和推送也一样变得很简单: $ git clone git@gitserver:/opt/git/project.git $ vim README $ git commit -am 'fix for the README file' $ git push origin master 用这个方法可以很快捷的为少数几个开发者架设一个可读写的 Git 服务。 作为一个额外的防范措施,你可以用 Git 自带的 git-shell 简单工具来把 git 用户的 活动限制在仅与 Git 相关。把它设为 git 用户登入的 shell,那么该用户就不能拥有主机 正常的 shell 访问权。为了实现这一点,需要指明用户的登入shell 是 git-shell ,而不 是 bash 或者 csh。你可能得编辑 /etc/passwd 文件: $ sudo vim /etc/passwd 在文件末尾,你应该能找到类似这样的行: git:x:1000:1000::/home/git:/bin/sh 把 bin/sh 改为 /usr/bin/git-shell (或者用 which git-shell 查看它的位置)。该 行修改后的样子如下: git:x:1000:1000::/home/git:/usr/bin/git-shell 现在 git 用户只能用 SSH 连接来推送和获取 Git 仓库,而不能直接使用主机 shell。 尝试登录的话,你会看到下面这样的拒绝信息: $ ssh git@gitserver fatal: What do you think I am? A shell? (你以为我是个啥?shell吗?) Connection to gitserver closed. (gitserver 连接已断开。) 77 第4章 服务器上的 Git Scott Chacon Pro Git 4.5 公共访问 匿名的读取权限该怎么实现呢?也许除了内部私有的项目之外,你还需要托管一些开源项 目。抑或你使用一些自动化的服务器来进行编译,或者一些经常变化的服务器群组,而又不 想整天生成新的 SSH 密钥——总之,你需要简单的匿名读取权限。 或许对小型的配置来说最简单的办法就是运行一个静态 web 服务,把它的根目录设定为 Git 仓库所在的位置,然后开启本章第一节提到的 post-update 挂钩。这里继续使用之前 的例子。假设仓库处于 /opt/git 目录,主机上运行着 Apache 服务。重申一下,任何 web 服务程序都可以达到相同效果;作为范例,我们将用一些基本的 Apache 设定来展示大体需 要的步骤。 首先,开启挂钩: $ cd project.git $ mv hooks/post-update.sample hooks/post-update $ chmod a+x hooks/post-update 假如使用的 Git 版本小于 1.6,那 mv 命令可以省略—— Git 是从较晚的版本才开始在 挂钩实例的结尾添加 .sample 后缀名的。 post-update 挂钩是做什么的呢?其内容大致如下: $ cat .git/hooks/post-update #!/bin/sh exec git-update-server-info 意思是当通过 SSH 向服务器推送时,Git 将运行这个命令来更新 HTTP 获取所需的文 件。 其次,在 Apache 配置文件中添加一个 VirtualHost 条目,把根文件(译注: Documen- tRoot 参数)设定为 Git 项目的根目录。假定 DNS 服务已经配置好,会把 .gitserver 发 送到任何你所在的主机来运行这些: ServerName git.gitserver DocumentRoot /opt/git Order allow, deny allow from all 另外,需要把 /opt/git 目录的 Unix 用户组设定为 www-data ,这样 web 服务才可以 读取仓库内容,因为 Apache 运行 CGI 脚本的模块(默认)使用的是该用户: $ chgrp -R www-data /opt/git 78 Scott Chacon Pro Git 4.6节 网页界面 GitWeb 重启 Apache 之后,就可以通过项目的 URL 来克隆该目录下的仓库了。 $ git clone http://git.gitserver/project.git 这一招可以让你在几分钟内为相当数量的用户架设好基于 HTTP 的读取权限。另一个提供 非授权访问的简单方法是开启一个 Git 守护进程,不过这将要求该进程的常驻——下一节 将是想走这条路的人准备的。 4.6 网页界面 GitWeb 如今我们的项目已经有了读写和只读的连接方式,也许应该再架设一个简单的网页界 面使其更加可视化。为此,Git 自带了一个叫做 GitWeb 的 CGI 脚本。你可以在类似 http://git.kernel.org 这样的站点找到 GitWeb 的应用实例(见图 4.1)。 图 4.1: 基于网页的 GitWeb 用户界面 如果想知道项目的 GitWeb 长什么样,Git 自带了一个命令,可以在类似 lighttpd 或 webrick 这样轻量级的服务器程序上打开一个临时的实例。在 Linux 主机上通常都安装 了 lighttpd ,这时就可以在项目目录里输入 git instaweb 来运行它。如果使用的是 Mac ,Leopard 预装了 Ruby,所以 webrick 应该是最好的选择。使用 lighttpd 以外的程序来 启用 git instaweb, 可以通过它的 --httpd 选项来实现。 $ git instaweb --httpd=webrick [2009-02-21 10:02:21] INFO WEBrick 1.3.1 [2009-02-21 10:02:21] INFO ruby 1.8.6 (2008-03-03) [universal-darwin9.0] 79 第4章 服务器上的 Git Scott Chacon Pro Git 这会在 1234 端口开启一个 HTTPD 服务,随之在浏览器中显示该页。简单的很。需要关 闭服务的时候,只要使用相同命令的 --stop 选项就好了: $ git instaweb --httpd=webrick --stop 如果需要为团队或者某个开源项目长期的运行 web 界面,那么 CGI 脚本就要由正常的网 页服务来运行。一些 Linux 发行版可以通过 apt 或 yum 安装一个叫做 gitweb 的软件包, 不妨首先尝试一下。我们将快速的介绍一下手动安装 GitWeb 的流程。首先,你需要 Git 的源码,其中带有 GitWeb,并能生成 CGI 脚本: $ git clone git://git.kernel.org/pub/scm/git/git.git $ cd git/ $ make GITWEB_PROJECTROOT="/opt/git" \ prefix=/usr gitweb/gitweb.cgi $ sudo cp -Rf gitweb /var/www/ 注意通过指定 GITWEB_PROJECTROOT 变量告诉编译命令 Git 仓库的位置。然后,让 Apache 来提供脚本的 CGI,为此添加一个 VirtualHost: ServerName gitserver DocumentRoot /var/www/gitweb Options ExecCGI +FollowSymLinks +SymLinksIfOwnerMatch AllowOverride All order allow,deny Allow from all AddHandler cgi-script cgi DirectoryIndex gitweb.cgi 不难想象,GitWeb 可以使用任何兼容 CGI 的网页服务来运行;如果偏向使用其他的(译 注:这里指Apache 以外的服务),配置也不会很麻烦。现在,通过 http://gitserver 就 可以在线访问仓库了,在 http://git.server 上还可以通过 HTTP 克隆和获取仓库的内 容。 Again, GitWeb can be served with any CGI capable web server; if you prefer to use something else, it shouldn’t be difficult to set up. At this point, you should be able to visit http://gitserver/ to view your repositories online, and you can use http://git.gitserver to clone and fetch your repositories over HTTP. 4.7 权限管理器 Gitosis 把所有用户的公钥保存在 authorized_keys 文件的做法只能暂时奏效。当用户数量到了 几百人的时候,它会变成一种痛苦。每一次都必须进入服务器的 shell,而且缺少对连接的 限制——文件里的每个人都对所有项目拥有读写权限。 80 Scott Chacon Pro Git 4.7节 权限管理器 Gitosis 现在,是时候向广泛使用的软件 Gitosis 求救了。Gitosis 简单的说就是一套用来管理 authorized_keys 文件和实现简单连接限制的脚本。最有意思的是,该软件用来添加用户和 设定权限的界面不是网页,而是一个特殊的 Git 仓库。你只需要设定好某个项目;然后推 送,Gitosis 就会随之改变服务器设定,酷吧? Gitosis 的安装算不上傻瓜化,不过也不算太难。用 Linux 服务器架设起来最简单—— 以下例子中的服务器使用 Ubuntu 8.10 系统。 Gitosis 需要使用部分 Python 工具,所以首先要安装 Python 的 setuptools 包,在 Ubuntu 中名为 python-setuptools: $ apt-get install python-setuptools 接下来,从项目主页克隆和安装 Gitosis: $ git clone git://eagain.net/gitosis.git $ cd gitosis $ sudo python setup.py install 这会安装几个 Gitosis 用的可执行文件。现在,Gitosis 想把它的仓库放在 /home/git, 倒也可以。不过我们的仓库已经建立在 /opt/git 了,这时可以创建一个文件连接,而不用 从头开始重新配置: $ ln -s /opt/git /home/git/repositories Gitosis 将为我们管理公钥,所以当前的文件需要删除,以后再重新添加公钥,并且让 Gitosis 自动控制 authorized_keys 文件。现在,把 authorized_keys文件移走: $ mv /home/git/.ssh/authorized_keys /home/git/.ssh/ak.bak 然后恢复 ‘git’ 用户的 shell,假设之前把它改成了 git-shell 命令。其他人仍然不 能通过它来登录系统,不过这次有 Gitosis 帮我们实现。所以现在把 /etc/passwd 文件的 这一行 git:x:1000:1000::/home/git:/usr/bin/git-shell 恢复成: git:x:1000:1000::/home/git:/bin/sh 81 第4章 服务器上的 Git Scott Chacon Pro Git 现在就可以初始化 Gitosis 了。需要通过自己的公钥来运行 gitosis-init。如果公钥不 在服务器上,则必须复制一份: $ sudo -H -u git gitosis-init < /tmp/id_dsa.pub Initialized empty Git repository in /opt/git/gitosis-admin.git/ Reinitialized existing Git repository in /opt/git/gitosis-admin.git/ 这样该公钥的拥有者就能修改包含着 Gitosis 设置的那个 Git 仓库了。然后手动将这个 新的控制仓库中的 post-update 脚本加上执行权限。 $ sudo chmod 755 /opt/git/gitosis-admin.git/hooks/post-update 万事俱备了。如果设定过程没出什么差错,现在可以试一下用初始化 Gitosis 公钥的拥 有者身份 SSH 进服务器。看到的结果应该和下面类似: $ ssh git@gitserver PTY allocation request failed on channel 0 fatal: unrecognized command 'gitosis-serve schacon@quaternion' Connection to gitserver closed. 说明 Gitosis 认出了该用户的身份,但由于没有运行任何 Git 命令所以它切断了连接。 所以,现在运行一个确切的 Git 命令——克隆 Gitosis 的控制仓库: # 在自己的电脑上 $ git clone git@gitserver:gitosis-admin.git 得到一个名为 gitosis-admin 的目录,主要由两部分组成: $ cd gitosis-admin $ find . ./gitosis.conf ./keydir ./keydir/scott.pub gitosis.conf 文件是用来设置用户、仓库和权限的控制文件。keydir 目录则是保存 所有具有访问权限用户公钥的地方——每人一个。你 keydir 中的文件名(前例中的 scott.pub)应该有所不同—— Gitosis 从使用 gitosis-init 脚本导入的公钥尾部的描述 中获取该名。 看一下 gitosis.conf 的内容,它应该只包含与刚刚克隆的 gitosis-admin 相关的信 息: 82 Scott Chacon Pro Git 4.7节 权限管理器 Gitosis $ cat gitosis.conf [gitosis] [group gitosis-admin] writable = gitosis-admin members = scott 它显示用户 scott ——初始化 Gitosis 公钥的拥有者——是唯一能访问 gitosis-admin 项目的人。 现在我们添加一个新的项目。我们将添加一个名为 mobile 的新节段,在这里罗列手机开 发团队的开发者以及他们需要访问权限的项目。由于 ‘scott’ 是系统中的唯一用户,我 们把它加成唯一的用户,从创建一个叫做 iphone_project 的新项目开始: [group mobile] writable = iphone_project members = scott 一旦修改了 gitosis-admin 项目的内容,只有提交并推送至服务器才能使之生效: $ git commit -am 'add iphone_project and mobile group' [master]: created 8962da8: "changed name" 1 files changed, 4 insertions(+), 0 deletions(-) $ git push Counting objects: 5, done. Compressing objects: 100% (2/2), done. Writing objects: 100% (3/3), 272 bytes, done. Total 3 (delta 1), reused 0 (delta 0) To git@gitserver:/opt/git/gitosis-admin.git fb27aec..8962da8 master -> master 第一次向新工程 iphone_project 的推送需要在本地的版本中把服务器添加为一个 remote 然后推送。从此手动为新项目在服务器上创建纯仓库的麻烦就是历史了—— Gitosis 会在 第一次遇到推送的时候自动创建它们: $ git remote add origin git@gitserver:iphone_project.git $ git push origin master Initialized empty Git repository in /opt/git/iphone_project.git/ Counting objects: 3, done. Writing objects: 100% (3/3), 230 bytes, done. Total 3 (delta 0), reused 0 (delta 0) To git@gitserver:iphone_project.git * [new branch] master -> master 注意到路径被忽略了(加上它反而没用),只有一个冒号加项目的名字—— Gitosis 会 为你找到项目的位置。 83 第4章 服务器上的 Git Scott Chacon Pro Git 要和朋友们共同在一个项目上共同工作,就得重新添加他们的公钥。不过这次不用在服务 器上一个一个手动添加到 ~/.ssh/authorized_keys 文件末端,而是在 keydir 目录为每一 个公钥添加一个文件。文件的命名将决定在 gitosis.conf 文件中用户的称呼。现在我们为 John,Josie 和 Jessica 添加公钥: $ cp /tmp/id_rsa.john.pub keydir/john.pub $ cp /tmp/id_rsa.josie.pub keydir/josie.pub $ cp /tmp/id_rsa.jessica.pub keydir/jessica.pub 然后把他们都加进 ‘mobile’ 团队,让他们对 iphone_project 具有读写权限: [group mobile] writable = iphone_project members = scott john josie jessica 如果你提交并推送这个修改,四个用户将同时具有该项目的读写权限。 Gitosis 也具有简单的访问控制功能。如果想让 John 只有读权限,可以这样做: [group mobile] writable = iphone_project members = scott josie jessica [group mobile_ro] readonly = iphone_project members = john 现在 John 可以克隆和获取更新,但 Gitosis 不会允许他向项目推送任何内容。这样的 组可以有尽可能有随意多个,每一个包含不同的用户和项目。甚至可以指定某个组为成员, 来继承它所有的成员。 如果出现了什么问题,把 loglevel=DEBUG 加入到 [gitosis] 部分或许有帮助(译注: 把日志设置到调试级别,记录更详细的信息)。如果你一不小心搞错了配置,失去了推送 权限,可以手动修改服务器上的 /home/git/.gitosis 文件—— Gitosis 从该文件读取信 息。一次推送会把 gitosis.conf 保存在服务器上。如果你手动编辑该文件,它将在你下次 向 gitosis-admin 推送之前它将保持原样。 4.8 Git 进程 公共,非授权的只读访问要求我们在 HTTP 协议的基础上使用 Git 协议。主因在于速 度。Git 协议更为高效,进而比 HTTP 协议更迅速,所以它能节省很多时间。 重申一下,这一点只适用于非授权、只读的访问。如果在防火墙之外的服务器上,该服务 的使用应该局限于公诸于世的项目。假如是在防火墙之内,它也可以用于具有大量参与人员 或者主机(长期整合资源或编译的服务器)的只读访问的项目,可以省去为逐一添加 SSH 公钥的麻烦。 84 Scott Chacon Pro Git 4.8节 Git 进程 无论哪种情况,Git 协议的设定都相对简单。基本上,只要以长期守护进程的形式运行该 命令: git daemon --reuseaddr --base-path=/opt/git/ /opt/git/ --reuseaddr 使得服务无须等到旧的连接尝试过期以后再重启,--base-path 选项使得克 隆项目的时候不用给出完整的路径,而最后面的路径告诉 Git 进程导出仓库的位置。假如 有防火墙,则需要为该主机的 9418 端口打个允许通信的洞。 有几个不同的办法可以让该进程长期驻留,取决于不同的操作系统。在 Ubuntu 主机上, 可以用 Upstart 脚本来完成。于是,在下面这个文件 /etc/event.d/local-git-daemon 加入该脚本内容: start on startup stop on shutdown exec /usr/bin/git daemon \ --user=git --group=git \ --reuseaddr \ --base-path=/opt/git/ \ /opt/git/ respawn 出于安全考虑,强烈建议用一个对仓库只有读取权限的用户身份来运行该进程——只需 要简单的新创建一个 git-ro 用户(译注:并将它对仓库的权限设为只读),用它来运行进 程。为了简化,下面我们将依旧使用运行了 Gitosis 的 ‘git’ 用户。 重启主机的时候,Git 进程会自行启动,一旦关闭了也会自行重启。要不重启就开启它, 可以运行这个命令: initctl start local-git-daemon 在其他系统上,或许应该使用 xinetd,sysinit 的一个脚本,或者其他的——只要能让 那个命令进程化和可监控。 然后,必须告诉 Gitosis 服务那些仓库允许基于 Git 协议的非授权访问。如果为每一个 仓库设立了自己的节段,就可以指定想让 Git 进程给予可读权限的仓库。假如要允许通过 Git 协议访问前面的 iphone 项目,可以把如下内容加到 gitosis.conf 文件的结尾: [repo iphone_project] daemon = yes 85 第4章 服务器上的 Git Scott Chacon Pro Git 在提交和推送完成以后,运行中的进程将开始相应所有能访问主机 9418 端口的人发来的 项目请求。 假如不想使用 Gitosis,而又想架设一个 Git 协议进程,则必须为每一个想使用 Git 进 程的项目运行如下命令: $ cd /path/to/project.git $ touch git-daemon-export-ok 该文件(译注:指空文件 git-deamon-export-ok)告诉 Git 允许对该项目的非授权访 问。 Gitosis 还能控制 GitWeb 显示哪些项目。首先,在 /etc/gitweb.conf 添加如下内容: $projects_list = "/home/git/gitosis/projects.list"; $projectroot = "/home/git/repositories"; $export_ok = "git-daemon-export-ok"; @git_base_url_list = ('git://gitserver'); 通过在 Gitosis 的设置文件里添加或删除 gitweb 设定,就能控制 GitWeb 允许用户浏 览哪些项目。比如,我们想让 iphone 项目在 GitWeb 里出现,把 repo 的设定改成下面的 样子: [repo iphone_project] daemon = yes gitweb = yes 如果现在提交和推送该项目,GitWeb 会自动开始展示我们的 iphone 项目。 4.9 Git 托管服务 如果不想经历自己架设 Git 服务器的麻烦,网络上有几个专业的仓库托管服务可供选 择。这样做有几大优点:托管账户的建立通常比较省时,方便项目的启动,而且不涉及服务 其的维护和监控。即使内部创建并运行了自己的服务器,为开源的代码使用一个公共托管站 点还是有好处——让开源社区更方便的找到该项目并给予帮助。 目前,可供选择的托管服务数量繁多,各有利弊。在 Git 官方 wiki 上的 Githosting 页面有一个持续更新的托管服务列表: http://git.or.cz/gitwiki/GitHosting 由于本书无法全部一一介绍它们,而本人(译注:指本书作者 Scott Chacon )刚好在其 中之一工作,我们将在这一节介绍一下在 GitHub 建立账户和开启新项目的过程。为你提供 一个使用托管服务的大致印象。 GitHub 是到目前为止最大的开源 Git 托管服务,并且是少数同时提供公共托管和私人托 管服务的站点之一,所以你可以在一个站点同时保存开源和商业代码。事实上,本书正是私 下使用 GitHub 合写的。(译注:而本书的翻译也是在 GitHub 上进行公共合作的)。 86 Scott Chacon Pro Git 4.9节 Git 托管服务 4.9.1 GitHub GitHub 和大多数的代码托管站点在处理项目命名空间的方式上略有不同。GitHub 的设计 更侧重于用户,而不是而不是全部基于项目。意谓本人在 GitHub 上托管一个 grit 项目的 话,它将不会出现在 github.com/grit,而是在 github.com/shacon/grit (译注:作者在 GitHub 上的用户名是 shacon)。不存在所谓某个项目的官方版本,所以假如第一作者放弃 了某个项目,它可以无缝转移到其它用户的旗下。 GitHub 同时也是一个向使用私有仓库的用户收取费用的商业公司,不过所有人都可以快 捷的得到一个免费账户并且在上面托管任意多的开源项目。我们将快速介绍一下该过程。 4.9.2 建立账户 第一个必要必要步骤是注册一个免费的账户。访问 Pricing and Signup (价格与注册) 页面 http://github.com/plans 并点击 Free acount (免费账户)的 “Sign Up(注册)” 按钮(见图 4.2),进入注册页面。 The first thing you need to do is set up a free user account. If you visit the Pricing and Signup page at http://github.com/plans and click the “Sign Up” button on the Free account (see figure 4-2), you’re taken to the signup page. 图 4.2: GitHub 服务简介页面 这里要求选择一个系统中尚未存在的用户名,提供一个与之相连的电邮地址,以及一个密 码(见图 4.3)。 如果事先有准备,可以顺便提供 SSH 公钥。我们在前文中的“小型安装” 一节介绍过生 成新公钥的方法。把生成的钥匙对中的公钥粘贴到 SSH Public Key (SSH 公钥)文本框 中。点击 “explain ssh keys” 链接可以获取在所有主流操作系统上完成该步骤的介绍。 点击 “I agree,sign me up (同意条款,让我注册)” 按钮就能进入新用户的控制面板 (见图 4.4)。 然后就可以建立新仓库了。 4.9.3 建立新仓库 点击用户面板上仓库旁边的 “create a new one(新建)” 连接。进入 Create a New Repository (新建仓库)表格(见图 4.5)。 唯一必做的仅仅是提供一个项目名称,当然也可以添加一点描述。搞定这些以后,点 “Create Repository(建立仓库)” 按钮。新仓库就建立起来了(见图4-6)。 87 第4章 服务器上的 Git Scott Chacon Pro Git 图 4.3: The GitHub user signup form 图 4.4: GitHub 用户面板 由于还没有提交代码,GitHub 会展示如何创建一个新项目,如何推送一个现存项目,以 及如何从一个公共的 Subversion 仓库导入项目(译注:这简直是公开挖 google code 和 sourceforge 的墙角)(见图 4.7)。 该指南和本书前文中的介绍类似。要把一个非 Git 项目变成 Git 项目,运行 $ git init $ git add . $ git commit -m 'initial commit' 一旦拥有一个本地 Git 仓库,把 GitHub 添加为远程仓库并推送 master 分支: 88 Scott Chacon Pro Git 4.9节 Git 托管服务 图 4.5: 在 GitHub 建立新仓库 图 4.6: GitHub 项目头信息 $ git remote add origin git@github.com:testinguser/iphone_project.git $ git push origin master 这时该项目就托管在 GitHub 上了。你可以把它的 URL 发给每个希望分享该工程的人。 本例的 URL 是 http://github.com/testinguser/iphone_project。你将在项目页面的头部 发现有两个 Git URL(见图 4.8)。 Public Clone URL(公共克隆 URL)是一个公开的,只读的 Git URL,任何人都可以通过 它克隆该项目。可以随意的散播这个 URL,发步到个人网站之类的地方。 Your Clone URL(私用克隆 URL)是一个给予 SSH 的读写 URL,只有使用与上传的 SSH 公钥对应的密钥来连接时,才能通过它进行读写操作。其他用户访问项目页面的时候看不到 该URL——只有公共的那个。 4.9.4 从 Subversion 中导入项目 如果想把某个公共 Subversion 项目导入 Git,GitHub 可以帮忙。在指南的最后有一个 指向导入 Subversion 页面的链接。点击它,可以得到一个表格,它包含着有关导入流程的 信息以及一个用来粘贴公共 Subversion 项目连接的文本框(见图 4.9)。 如果项目很大,采用非标准结构,或者是私有的,那么该流程将不适用。在第七章,你将 了解到手动导入复杂工程的方法。 89 第4章 服务器上的 Git Scott Chacon Pro Git 图 4.7: 新仓库指南 图 4.8: 项目开头的公共 URL 和私有 URL 。 4.9.5 开始合作 现在把团队里其他的人也加进来。如果 John,Josie 和 Jessica 都在 GitHub 注册了账 户,要给他们向仓库推送的访问权,可以把它们加为项目合作者。这样他们的公钥就能用来 向仓库推送了。 点击项目页面上方的 “edit(编辑)” 按钮或者顶部的 Admin (管理)标签进入项目 管理页面(见图 4.10)。 为了给另一个用户添加项目的写权限,点击 “Add another collaborator(添加另一个 合作者)” 链接。一个新文本框会出现,用来输入用户名。在输入用户名的同时将会跳出 一个帮助提示,显示出可能匹配的用户名。找到正确的用户名以后,点 Add (添加)按 钮,把它变成该项目的合作者(见图 4.11)。 添加完合作者以后,就可以在 Repository Collaborators (仓库合作者)区域看到他们 的列表(见图 4.12)。 如果需要取消某人的访问权,点击 “revoke (撤销)”,他的推送权限就被删除了。在 未来的项目中,可以通过复制现存项目的权限设定来得到相同的合作者群组。 90 Scott Chacon Pro Git 4.9节 Git 托管服务 图 4.9: Subversion 导入界面 图 4.10: GitHub 管理页面 图 4.11: 为项目添加合作者 4.9.6 项目页面 在推送或从 Subversion 导入项目之后,你会得到一个类似图 4.13 的项目主页。 其他人访问你的项目时,他们会看到该页面。它包含了该项目不同方面的标签。Commits 标签将按时间展示逆序的 commit 列表,与 git log 命令的输出类似。Network 标签展示 所有 fork 了该项目并做出贡献的用户的关系图。Downloads 标签允许你上传项目的二进制 文件,并提供了指向该项目所有标记过的位置的 tar/zip 打包下载连接。Wiki 标签提供 了一个用来撰写文档或其他项目相关信息的 wiki。Graphs 标签包含了一些可视化的项目 91 第4章 服务器上的 Git Scott Chacon Pro Git 图 4.12: 项目合作者列表 图 4.13: GitHub 项目主页 信息与数据。刚开始进入的 Source 标签页面列出了项目的主目录;并且在下方自动展示 README 文件的内容(如果该文件存在的话)。该标签还包含了最近一次提交的相关信息。 4.9.7 派生(forking)项目 如果想向一个自己没有推送权限的项目贡献代码,GitHub 提倡使用派生(forking)。在 你发现一个感兴趣的项目,打算在上面 Hack 一把的时候,可以点击页面上方的 “fork(派 生)” 按钮,GitHub 会为你的用户复制一份该项目,这样你就可以向它推送内容了。 使用这个办法,项目维护者不用操心为了推送权限把其他人加为合作者的麻烦。大家可以 派生一个项目副本并进行推送,而后项目的主要维护者可以把这些副本添加为远程仓库,从 中拉取更新的内容进行合并。 92 Scott Chacon Pro Git 4.10节 小节 要派生一个项目,到该项目的页面(本例中是 mojombo/chronic)点击上面的 “fork” 按钮(见图 4.14)。 图 4.14: 点击 “fork” 按钮来获得任意项目的可写副本 几秒钟以后,你将进入新建的项目页面,显示出该项目是派生自另一个项目的副本(见图 4.15)。 图 4.15: 你派生的项目副本 4.9.8 GitHub 小节 GitHub 就介绍这么多,不过意识到做到这些是多么快捷十分重要。不过几分钟的时间, 你就能创建一个账户,添加一个新的项目并开始推送。如果你的项目是开源的,它还同时获 得了对庞大的开发者社区的可视性,社区成员可能会派生它并做出贡献。退一万步讲,这至 少是个快速开始尝试 Git 的好办法。 4.10 小节 几个不同的方案可以让你获得远程 Git 仓库来与其他人合作或分享你的成果。 运行自己的服务器意味着更多的控制权以及在防火墙内部操作的可能性,然而这样的服务 器通常需要投入一定的时间来架设和维护。如果把数据放在托管服务上,假设和维护变得十 分简单;然而,你不得不把代码保存在别人的服务器上,很多公司不允许这种做法。 使用哪个方案或哪种方案的组合对你和你的团队更合适,应该不是一个太难的决定。 93 第5章 分布式 Git 为了便于项目中的所有开发者分享代码,我们准备好了一台服务器存放远程 Git 仓库。 经过前面几章的学习,我们已经学会了一些基本的本地工作流程中所需用到的命令。接下 来,我们要学习下如何利用 Git 来组织和完成分布式工作流程。 特别是,当作为项目贡献者时,我们该怎么做才能方便维护者采纳更新;或者作为项目维 护者时,又该怎样有效管理大量贡献者的提交。 5.1 分布式工作流程 同传统的集中式版本控制系统(CVCS)不同,开发者之间的协作方式因着 Git 的分布式 特性而变得更为灵活多样。在集中式系统上,每个开发者就像是连接在集线器上的节点,彼 此的工作方式大体相像。而在 Git 网络中,每个开发者同时扮演着节点和集线器的角色, 这就是说,每一个开发者都可以将自己的代码贡献到另外一个开发者的仓库中,或者建立自 己的公共仓库,让其他开发者基于自己的工作开始,为自己的仓库贡献代码。于是,Git 的 分布式协作便可以衍生出种种不同的工作流程,我会在接下来的章节介绍几种常见的应用方 式,并分别讨论各自的优缺点。你可以选择其中的一种,或者结合起来,应用到你自己的项 目中。 5.1.1 集中式工作流 通常,集中式工作流程使用的都是单点协作模型。一个存放代码仓库的中心服务器,可以 接受所有开发者提交的代码。所有的开发者都是普通的节点,作为中心集线器的消费者,平 时的工作就是和中心仓库同步数据(见图 5.1)。 图 5.1: 集中式工作流 95 第5章 分布式 Git Scott Chacon Pro Git 如果两个开发者从中心仓库克隆代码下来,同时作了一些修订,那么只有第一个开发者可 以顺利地把数据推送到共享服务器。第二个开发者在提交他的修订之前,必须先下载合并 服务器上的数据,解决冲突之后才能推送数据到共享服务器上。在 Git 中这么用也决无问 题,这就好比是在用 Subversion(或其他 CVCS)一样,可以很好地工作。 如果你的团队不是很大,或者大家都已经习惯了使用集中式工作流程,完全可以采用这种 简单的模式。只需要配置好一台中心服务器,并给每个人推送数据的权限,就可以开展工 作了。但如果提交代码时有冲突, Git 根本就不会让用户覆盖他人代码,它直接驳回第二 个人的提交操作。这就等于告诉提交者,你所作的修订无法通过快近(fast-forward)来合 并,你必须先拉取最新数据下来,手工解决冲突合并后,才能继续推送新的提交。绝大多数 人都熟悉和了解这种模式的工作方式,所以使用也非常广泛。 5.1.2 集成管理员工作流 由于 Git 允许使用多个远程仓库,开发者便可以建立自己的公共仓库,往里面写数据并 共享给他人,而同时又可以从别人的仓库中提取他们的更新过来。这种情形通常都会有个代 表着官方发布的项目仓库(blessed repository),开发者们由此仓库克隆出一个自己的公 共仓库(developer public),然后将自己的提交推送上去,请求官方仓库的维护者拉取更 新合并到主项目。维护者在自己的本地也有个克隆仓库(integration manager),他可以 将你的公共仓库作为远程仓库添加进来,经过测试无误后合并到主干分支,然后再推送到官 方仓库。工作流程看起来就像图 5.2 所示: 1. 项目维护者可以推送数据到公共仓库 blessed repository。 2. 贡献者克隆此仓库,修订或编写新代码。 3. 贡献者推送数据到自己的公共仓库 developer public。 4. 贡献者给维护者发送邮件,请求拉取自己的最新修订。 5. 维护者在自己本地的 integration manger 仓库中,将贡献者的仓库加为远程仓库, 合并更新并做测试。 6. 维护者将合并后的更新推送到主仓库 blessed repository。 图 5.2: 集成管理员工作流 在 GitHub 网站上使用得最多的就是这种工作流。人们可以复制(fork 亦即克隆)某个 项目到自己的列表中,成为自己的公共仓库。随后将自己的更新提交到这个仓库,所有人都 可以看到你的每次更新。这么做最主要的优点在于,你可以按照自己的节奏继续工作,而不 必等待维护者处理你提交的更新;而维护者也可以按照自己的节奏,任何时候都可以过来处 理接纳你的贡献。 96 Scott Chacon Pro Git 5.2节 为项目作贡献 5.1.3 司令官与副官工作流 这其实是上一种工作流的变体。一般超大型的项目才会用到这样的工作方式,像是拥有数 百协作开发者的 Linux 内核项目就是如此。各个集成管理员分别负责集成项目中的特定部 分,所以称为副官(lieutenant)。而所有这些集成管理员头上还有一位负责统筹的总集成 管理员,称为司令官(dictator)。司令官维护的仓库用于提供所有协作者拉取最新集成的 项目代码。整个流程看起来如图 5.3 所示: 1. 一般的开发者在自己的特性分支上工作,并不定期地根据主干分支(dectator 上的 master)衍合。 2. 副官(lieutenant)将普通开发者的特性分支合并到自己的 master 分支中。 3. 司令官(dictator)将所有副官的 master 分支并入自己的 master 分支。 4. 司令官(dictator)将集成后的 master 分支推送到共享仓库 blessed repository 中,以便所有其他开发者以此为基础进行衍合。 图 5.3: 司令官与副官工作流 这种工作流程并不常用,只有当项目极为庞杂,或者需要多级别管理时,才会体现出优 势。利用这种方式,项目总负责人(即司令官)可以把大量分散的集成工作委托给不同的小 组负责人分别处理,最后再统筹起来,如此各人的职责清晰明确,也不易出错(译注:此乃 分而治之)。 以上介绍的是常见的分布式系统可以应用的工作流程,当然不止于 Git。在实际的开发工 作中,你可能会遇到各种为了满足特定需求而有所变化的工作方式。我想现在你应该已经清 楚,接下来自己需要用哪种方式开展工作了。下节我还会再举些例子,看看各式工作流中的 每个角色具体应该如何操作。 5.2 为项目作贡献 接下来,我们来学习一下作为项目贡献者,会有哪些常见的工作模式。 不过要说清楚整个协作过程真的很难,Git 如此灵活,人们的协作方式便可以各式各样, 没有固定不变的范式可循,而每个项目的具体情况又多少会有些不同,比如说参与者的规 模,所选择的工作流程,每个人的提交权限,以及 Git 以外贡献等等,都会影响到具体操 作的细节。 97 第5章 分布式 Git Scott Chacon Pro Git 首当其冲的是参与者规模。项目中有多少开发者是经常提交代码的?经常又是多久呢?大 多数两至三人的小团队,一天大约只有几次提交,如果不是什么热门项目的话就更少了。可 要是在大公司里,或者大项目中,参与者可以多到上千,每天都会有十几个上百个补丁提交 上来。这种差异带来的影响是显著的,越是多的人参与进来,就越难保证每次合并正确无 误。你正在工作的代码,可能会因为合并进来其他人的更新而变得过时,甚至受创无法运 行。而已经提交上去的更新,也可能在等着审核合并的过程中变得过时。那么,我们该怎样 做才能确保代码是最新的,提交的补丁也是可用的呢? 接下来便是项目所采用的工作流。是集中式的,每个开发者都具有等同的写权限?项目是 否有专人负责检查所有补丁?是不是所有补丁都做过同行复阅(peer-review)再通过审核 的?你是否参与审核过程?如果使用副官系统,那你是不是限定于只能向此副官提交? 还有你的提交权限。有或没有向主项目提交更新的权限,结果完全不同,直接决定最终采 用怎样的工作流。如果不能直接提交更新,那该如何贡献自己的代码呢?是不是该有个什么 策略?你每次贡献代码会有多少量?提交频率呢? 所有以上这些问题都会或多或少影响到最终采用的工作流。接下来,我会在一系列由简入 繁的具体用例中,逐一阐述。此后在实践时,应该可以借鉴这里的例子,略作调整,以满足 实际需要构建自己的工作流。 5.2.1 提交指南 开始分析特定用例之前,先来了解下如何撰写提交说明。一份好的提交指南可以帮助 协作者更轻松更有效地配合。Git 项目本身就提供了一份文档(Git 项目源代码目录中 Documentation/SubmittingPatches),列数了大量提示,从如何编撰提交说明到提交补 丁,不一而足。 首先,请不要在更新中提交多余的白字符(whitespace)。Git 有种检查此类问题的方 法,在提交之前,先运行 git diff --check,会把可能的多余白字符修正列出来。下面的 示例,我已经把终端中显示为红色的白字符用 X 替换掉: $ git diff --check lib/simplegit.rb:5: trailing whitespace. + @git_dir = File.expand_path(git_dir)XX lib/simplegit.rb:7: trailing whitespace. + XXXXXXXXXXX lib/simplegit.rb:26: trailing whitespace. + def command(git_cmd)XXXX 这样在提交之前你就可以看到这类问题,及时解决以免困扰其他开发者。 接下来,请将每次提交限定于完成一次逻辑功能。并且可能的话,适当地分解为多次小更 新,以便每次小型提交都更易于理解。请不要在周末穷追猛打一次性解决五个问题,而最后 拖到周一再提交。就算是这样也请尽可能利用暂存区域,将之前的改动分解为每次修复一个 问题,再分别提交和加注说明。如果针对两个问题改动的是同一个文件,可以试试看 git add --patch 的方式将部分内容置入暂存区域(我们会在第六章再详细介绍)。无论是五次 小提交还是混杂在一起的大提交,最终分支末端的项目快照应该还是一样的,但分解开来之 后,更便于其他开发者复阅。这么做也方便自己将来取消某个特定问题的修复。我们将在第 六章介绍一些重写提交历史,同暂存区域交互的技巧和工具,以便最终得到一个干净有意 义,且易于理解的提交历史。 98 Scott Chacon Pro Git 5.2节 为项目作贡献 最后需要谨记的是提交说明的撰写。写得好可以让大家协作起来更轻松。一般来说,提交 说明最好限制在一行以内,50 个字符以下,简明扼要地描述更新内容,空开一行后,再展 开详细注解。Git 项目本身需要开发者撰写详尽注解,包括本次修订的因由,以及前后不 同实现之间的比较,我们也该借鉴这种做法。另外,提交说明应该用祈使现在式语态,比 如,不要说成 “I added tests for” 或 “Adding tests for” 而应该用 “Add tests for”。下面是来自 tpope.net 的 Tim Pope 原创的提交说明格式模版,供参考: 本次更新的简要描述(50 个字符以内) 如果必要,此处展开详尽阐述。段落宽度限定在 72 个字符以内。 某些情况下,第一行的简要描述将用作邮件标题,其余部分作为邮件正文。 其间的空行是必要的,以区分两者(当然没有正文另当别论)。 如果并在一起,rebase 这样的工具就可能会迷惑。 另起空行后,再进一步补充其他说明。 - 可以使用这样的条目列举式。 - 一般以单个空格紧跟短划线或者星号作为每项条目的起始符。每个条目间用一空行隔开。 不过这里按自己项目的约定,可以略作变化。 如果你的提交说明都用这样的格式来书写,好多事情就可以变得十分简单。Git 项目本身 就是这样要求的,我强烈建议你到 Git 项目仓库下运行 git log --no-merges 看看,所 有提交历史的说明是怎样撰写的。(译注:如果现在还没有克隆 git 项目源代码,是时候 git clone git://git.kernel.org/pub/scm/git/git.git 了。) 为简单起见,在接下来的例子(及本书随后的所有演示)中,我都不会用这种格式,而使 用 -m 选项提交 git commit。不过请还是按照我之前讲的做,别学我这里偷懒的方式。 5.2.2 私有的小型团队 我们从最简单的情况开始,一个私有项目,与你一起协作的还有另外一到两位开发者。这 里说私有,是指源代码不公开,其他人无法访问项目仓库。而你和其他开发者则都具有推送 数据到仓库的权限。 这种情况下,你们可以用 Subversion 或其他集中式版本控制系统类似的工作流来协作。 你仍然可以得到 Git 带来的其他好处:离线提交,快速分支与合并等等,但工作流程还是 差不多的。主要区别在于,合并操作发生在客户端而非服务器上。让我们来看看,两个开发 者一起使用同一个共享仓库,会发生些什么。第一个人,John,克隆了仓库,作了些更新, 在本地提交。(下面的例子中省略了常规提示,用 ... 代替以节约版面。) # John's Machine $ git clone john@githost:simplegit.git Initialized empty Git repository in /home/john/simplegit/.git/ ... $ cd simplegit/ $ vim lib/simplegit.rb $ git commit -am 'removed invalid default value' [master 738ee87] removed invalid default value 1 files changed, 1 insertions(+), 1 deletions(-) 99 第5章 分布式 Git Scott Chacon Pro Git 第二个开发者,Jessica,一样这么做:克隆仓库,提交更新: # Jessica's Machine $ git clone jessica@githost:simplegit.git Initialized empty Git repository in /home/jessica/simplegit/.git/ ... $ cd simplegit/ $ vim TODO $ git commit -am 'add reset task' [master fbff5bc] add reset task 1 files changed, 1 insertions(+), 0 deletions(-) 现在,Jessica 将她的工作推送到服务器上: # Jessica's Machine $ git push origin master ... To jessica@githost:simplegit.git 1edee6b..fbff5bc master -> master John 也尝试推送自己的工作上去: # John's Machine $ git push origin master To john@githost:simplegit.git ! [rejected] master -> master (non-fast forward) error: failed to push some refs to 'john@githost:simplegit.git' John 的推送操作被驳回,因为 Jessica 已经推送了新的数据上去。请注意,特别是你 用惯了 Subversion 的话,这里其实修改的是两个文件,而不是同一个文件的同一个地 方。Subversion 会在服务器端自动合并提交上来的更新,而 Git 则必须先在本地合并后才 能推送。于是,John 不得不先把 Jessica 的更新拉下来: $ git fetch origin ... From john@githost:simplegit + 049d078...fbff5bc master -> origin/master 此刻,John 的本地仓库如图 5.4 所示: 虽然 John 下载了 Jessica 推送到服务器的最近更新(fbff5),但目前只是 origin/ master 指针指向它,而当前的本地分支 master 仍然指向自己的更新(738ee),所以需要 先把她的提交合并过来,才能继续推送数据: 100 Scott Chacon Pro Git 5.2节 为项目作贡献 图 5.4: John 的仓库历史 $ git merge origin/master Merge made by recursive. TODO | 1 + 1 files changed, 1 insertions(+), 0 deletions(-) 还好,合并过程非常顺利,没有冲突,现在 John 的提交历史如图 5.5 所示: 图 5.5: 合并 origin/master 后 John 的仓库历史 现在,John 应该再测试一下代码是否仍然正常工作,然后将合并结果(72bbc)推送到服 务器上: $ git push origin master ... To john@githost:simplegit.git fbff5bc..72bbc59 master -> master 最终,John 的提交历史变为图 5.6 所示: 而在这段时间,Jessica 已经开始在另一个特性分支工作了。她创建了 issue54 并提交 了三次更新。她还没有下载 John 提交的合并结果,所以提交历史如图 5.7 所示: 101 第5章 分布式 Git Scott Chacon Pro Git 图 5.6: 推送后 John 的仓库历史 图 5.7: Jessica 的提交历史 Jessica 想要先和服务器上的数据同步,所以先下载数据: # Jessica's Machine $ git fetch origin ... From jessica@githost:simplegit fbff5bc..72bbc59 master -> origin/master 于是 Jessica 的本地仓库历史多出了 John 的两次提交(738ee 和 72bbc),如图 5.8 所示: 图 5.8: 获取 John 的更新之后 Jessica 的提交历史 此时,Jessica 在特性分支上的工作已经完成,但她想在推送数据之前,先确认下要并进 来的数据究竟是什么,于是运行 git log 查看: $ git log --no-merges origin/master ^issue54 commit 738ee872852dfaa9d6634e0dea7a324040193016 Author: John Smith Date: Fri May 29 16:01:27 2009 -0700 removed invalid default value 102 Scott Chacon Pro Git 5.2节 为项目作贡献 现在,Jessica 可以将特性分支上的工作并到 master 分支,然后再并入 John 的工作 (origin/master)到自己的 master 分支,最后再推送回服务器。当然,得先切回主分支 才能集成所有数据: $ git checkout master Switched to branch "master" Your branch is behind 'origin/master' by 2 commits, and can be fast-forwarded. 要合并 origin/master 或 issue54 分支,谁先谁后都没有关系,因为它们都在上游 (upstream)(译注:想像分叉的更新像是汇流成河的源头,所以上游 upstream 是指最新 的提交),所以无所谓先后顺序,最终合并后的内容快照都是一样的,而仅是提交历史看起 来会有些先后差别。Jessica 选择先合并 issue54: $ git merge issue54 Updating fbff5bc..4af4298 Fast forward README | 1 + lib/simplegit.rb | 6 +++++- 2 files changed, 6 insertions(+), 1 deletions(-) 正如所见,没有冲突发生,仅是一次简单快进。现在 Jessica 开始合并 John 的工作 (origin/master): $ git merge origin/master Auto-merging lib/simplegit.rb Merge made by recursive. lib/simplegit.rb | 2 +- 1 files changed, 1 insertions(+), 1 deletions(-) 所有的合并都非常干净。现在 Jessica 的提交历史如图 5.9 所示: 图 5.9: 合并 John 的更新后 Jessica 的提交历史 现在 Jessica 已经可以在自己的 master 分支中访问 origin/master 的最新改动了, 所以她应该可以成功推送最后的合并结果到服务器上(假设 John 此时没再推送新数据上 来): 103 第5章 分布式 Git Scott Chacon Pro Git $ git push origin master ... To jessica@githost:simplegit.git 72bbc59..8059c15 master -> master 至此,每个开发者都提交了若干次,且成功合并了对方的工作成果,最新的提交历史如图 5.10 所示: 图 5.10: Jessica 推送数据后的提交历史 以上就是最简单的协作方式之一:先在自己的特性分支中工作一段时间,完成后合并到自 己的 master 分支;然后下载合并 origin/master 上的更新(如果有的话),再推回远程 服务器。一般的协作流程如图 5.11 所示: 5.2.3 私有团队间协作 现在我们来看更大一点规模的私有团队协作。如果有几个小组分头负责若干特性的开发和 集成,那他们之间的协作过程是怎样的。 假设 John 和 Jessica 一起负责开发某项特性 A,而同时 Jessica 和 Josie 一起负责 开发另一项功能 B。公司使用典型的集成管理员式工作流,每个组都有一名管理员负责集成 本组代码,及更新项目主仓库的 master 分支。所有开发都在代表小组的分支上进行。 让我们跟随 Jessica 的视角看看她的工作流程。她参与开发两项特性,同时和不同小 组的开发者一起协作。克隆生成本地仓库后,她打算先着手开发特性 A。于是创建了新的 featureA 分支,继而编写代码: # Jessica's Machine $ git checkout -b featureA Switched to a new branch "featureA" $ vim lib/simplegit.rb $ git commit -am 'add limit to log function' [featureA 3300904] add limit to log function 1 files changed, 1 insertions(+), 1 deletions(-) 此刻,她需要分享目前的进展给 John,于是她将自己的 featureA 分支提交到服务器。 由于 Jessica 没有权限推送数据到主仓库的 master 分支(只有集成管理员有此权限), 所以只能将此分支推上去同 John 共享协作: 104 Scott Chacon Pro Git 5.2节 为项目作贡献 图 5.11: 多用户共享仓库协作方式的一般工作流程时序 $ git push origin featureA ... To jessica@githost:simplegit.git * [new branch] featureA -> featureA Jessica 发邮件给 John 让他上来看看 featureA 分支上的进展。在等待他的反馈之 前,Jessica 决定继续工作,和 Josie 一起开发 featureB 上的特性 B。当然,先创建此 分支,分叉点以服务器上的 master 为起点: # Jessica's Machine $ git fetch origin 105 第5章 分布式 Git Scott Chacon Pro Git $ git checkout -b featureB origin/master Switched to a new branch "featureB" 随后,Jessica 在 featureB 上提交了若干更新: $ vim lib/simplegit.rb $ git commit -am 'made the ls-tree function recursive' [featureB e5b0fdc] made the ls-tree function recursive 1 files changed, 1 insertions(+), 1 deletions(-) $ vim lib/simplegit.rb $ git commit -am 'add ls-files' [featureB 8512791] add ls-files 1 files changed, 5 insertions(+), 0 deletions(-) 现在 Jessica 的更新历史如图 5.12 所示: 图 5.12: Jessica 的更新历史 Jessica 正准备推送自己的进展上去,却收到 Josie 的来信,说是她已经将自己的工作 推到服务器上的 featureBee 分支了。这样,Jessica 就必须先将 Josie 的代码合并到自 己本地分支中,才能再一起推送回服务器。她用 git fetch 下载 Josie 的最新代码: $ git fetch origin ... From jessica@githost:simplegit * [new branch] featureBee -> origin/featureBee 然后 Jessica 使用 git merge 将此分支合并到自己分支中: $ git merge origin/featureBee Auto-merging lib/simplegit.rb Merge made by recursive. lib/simplegit.rb | 4 ++++ 1 files changed, 4 insertions(+), 0 deletions(-) 106 Scott Chacon Pro Git 5.2节 为项目作贡献 合并很顺利,但另外有个小问题:她要推送自己的 featureB 分支到服务器上的 fea- tureBee 分支上去。当然,她可以使用冒号(:)格式指定目标分支: $ git push origin featureB:featureBee ... To jessica@githost:simplegit.git fba9af8..cd685d1 featureB -> featureBee 我们称此为refspec。更多有关于 Git refspec 的讨论和使用方式会在第九章作详细阐 述。 接下来,John 发邮件给 Jessica 告诉她,他看了之后作了些修改,已经推回服务器 featureA 分支,请她过目下。于是 Jessica 运行 git fetch 下载最新数据: $ git fetch origin ... From jessica@githost:simplegit 3300904..aad881d featureA -> origin/featureA 接下来便可以用 git log 查看更新了些什么: $ git log origin/featureA ^featureA commit aad881d154acdaeb2b6b18ea0e827ed8a6d671e6 Author: John Smith Date: Fri May 29 19:57:33 2009 -0700 changed log output to 30 from 25 最后,她将 John 的工作合并到自己的 featureA 分支中: $ git checkout featureA Switched to branch "featureA" $ git merge origin/featureA Updating 3300904..aad881d Fast forward lib/simplegit.rb | 10 +++++++++- 1 files changed, 9 insertions(+), 1 deletions(-) Jessica 稍做一番修整后同步到服务器: $ git commit -am 'small tweak' [featureA ed774b3] small tweak 1 files changed, 1 insertions(+), 1 deletions(-) $ git push origin featureA ... To jessica@githost:simplegit.git 3300904..ed774b3 featureA -> featureA 107 第5章 分布式 Git Scott Chacon Pro Git 现在的 Jessica 提交历史如图 5.13 所示: 图 5.13: 在特性分支中提交更新后的提交历史 现在,Jessica,Josie 和 John 通知集成管理员服务器上的 featureA 及 featureBee 分支已经准备好,可以并入主线了。在管理员完成集成工作后,主分支上便多出一个新的合 并提交(5399e),用 fetch 命令更新到本地后,提交历史如图 5.14 所示: 图 5.14: 合并特性分支后的 Jessica 提交历史 许多开发小组改用 Git 就是因为它允许多个小组间并行工作,而在稍后恰当时机再行合 并。通过共享远程分支的方式,无需干扰整体项目代码便可以开展工作,因此使用 Git 的 小型团队间协作可以变得非常灵活自由。以上工作流程的时序如图 5.15 所示: 5.2.4 公开的小型项目 上面说的是私有项目协作,但要给公开项目作贡献,情况就有些不同了。因为你没有直接 更新主仓库分支的权限,得寻求其它方式把工作成果交给项目维护人。下面会介绍两种方 法,第一种使用 git 托管服务商提供的仓库复制功能,一般称作 fork,比如 repo.or.cz 和 GitHub 都支持这样的操作,而且许多项目管理员都希望大家使用这样的方式。另一种方 法是通过电子邮件寄送文件补丁。 但不管哪种方式,起先我们总需要克隆原始仓库,而后创建特性分支开展工作。基本工作 流程如下: 108 Scott Chacon Pro Git 5.2节 为项目作贡献 图 5.15: 团队间协作工作流程基本时序 $ git clone (url) $ cd project $ git checkout -b featureA $ (work) $ git commit $ (work) $ git commit 你可能想到用 rebase -i 将所有更新先变作单个提交,又或者想重新安排提交之间的差 异补丁,以方便项目维护者审阅 — 有关交互式衍合操作的细节见第六章。 在完成了特性分支开发,提交给项目维护者之前,先到原始项目的页面上点击“Fork”按 钮,创建一个自己可写的公共仓库(译注:即下面的 url 部分,参照后续的例子,应该是 git://githost/simplegit.git)。然后将此仓库添加为本地的第二个远端仓库,姑且称为 myfork: 109 第5章 分布式 Git Scott Chacon Pro Git $ git remote add myfork (url) 你需要将本地更新推送到这个仓库。要是将远端 master 合并到本地再推回去,还不如把 整个特性分支推上去来得干脆直接。而且,假若项目维护者未采纳你的贡献的话(不管是直 接合并还是 cherry pick),都不用回退(rewind)自己的 master 分支。但若维护者合并 或 cherry-pick 了你的工作,最后总还可以从他们的更新中同步这些代码。好吧,现在先 把 featureA 分支整个推上去: $ git push myfork featureA 然后通知项目管理员,让他来抓取你的代码。通常我们把这件事叫做 pull request。可 以直接用 GitHub 等网站提供的 “pull request” 按钮自动发送请求通知;或手工把 git request-pull 命令输出结果电邮给项目管理员。 request-pull 命令接受两个参数,第一个是本地特性分支开始前的原始分支,第二个是 请求对方来抓取的 Git 仓库 URL(译注:即下面 myfork 所指的,自己可写的公共仓库)。 比如现在Jessica 准备要给 John 发一个 pull requst,她之前在自己的特性分支上提交了 两次更新,并把分支整个推到了服务器上,所以运行该命令会看到: $ git request-pull origin/master myfork The following changes since commit 1edee6b1d61823a2de3b09c160d7080b8d1b3a40: John Smith (1): added a new function are available in the git repository at: git://githost/simplegit.git featureA Jessica Smith (2): add limit to log function change log output to 30 from 25 lib/simplegit.rb | 10 +++++++++- 1 files changed, 9 insertions(+), 1 deletions(-) 输出的内容可以直接发邮件给管理者,他们就会明白这是从哪次提交开始旁支出去的,该 到哪里去抓取新的代码,以及新的代码增加了哪些功能等等。 像这样随时保持自己的 master 分支和官方 origin/master 同步,并将自己的工作限制 在特性分支上的做法,既方便又灵活,采纳和丢弃都轻而易举。就算原始主干发生变化,我 们也能重新衍合提供新的补丁。比如现在要开始第二项特性的开发,不要在原来已推送的特 性分支上继续,还是按原始 master 开始: $ git checkout -b featureB origin/master $ (work) 110 Scott Chacon Pro Git 5.2节 为项目作贡献 $ git commit $ git push myfork featureB $ (email maintainer) $ git fetch origin 现在,A、B 两个特性分支各不相扰,如同竹筒里的两颗豆子,队列中的两个补丁,你 随时都可以分别从头写过,或者衍合,或者修改,而不用担心特性代码的交叉混杂。如图 5.16 所示: 图 5.16: featureB 以后的提交历史 假设项目管理员接纳了许多别人提交的补丁后,准备要采纳你提交的第一个分支,却发 现因为代码基准不一致,合并工作无法正确干净地完成。这就需要你再次衍合到最新的 origin/master,解决相关冲突,然后重新提交你的修改: $ git checkout featureA $ git rebase origin/master $ git push -f myfork featureA 自然,这会重写提交历史,如图 5.17 所示: 图 5.17: featureA 重新衍合后的提交历史 注意,此时推送分支必须使用 -f 选项(译注:表示 force,不作检查强制重写)替换远 程已有的 featureA 分支,因为新的 commit 并非原来的后续更新。当然你也可以直接推送 到另一个新的分支上去,比如称作 featureAv2。 再考虑另一种情形:管理员看过第二个分支后觉得思路新颖,但想请你改下具体实现。我 们只需以当前 origin/master 分支为基准,开始一个新的特性分支 featureBv2,然后把原 来的 featureB 的更新拿过来,解决冲突,按要求重新实现部分代码,然后将此特性分支推 送上去: 111 第5章 分布式 Git Scott Chacon Pro Git $ git checkout -b featureBv2 origin/master $ git merge --no-commit --squash featureB $ (change implementation) $ git commit $ git push myfork featureBv2 这里的 --squash 选项将目标分支上的所有更改全拿来应用到当前分支上,而 --no- commit 选项告诉 Git 此时无需自动生成和记录(合并)提交。这样,你就可以在原来代码 基础上,继续工作,直到最后一起提交。 好了,现在可以请管理员抓取 featureBv2 上的最新代码了,如图 5.18 所示: 图 5.18: featureBv2 之后的提交历史 5.2.5 公开的大型项目 许多大型项目都会立有一套自己的接受补丁流程,你应该注意下其中细节。但多数项目都 允许通过开发者邮件列表接受补丁,现在我们来看具体例子。 整个工作流程类似上面的情形:为每个补丁创建独立的特性分支,而不同之处在于如何提 交这些补丁。不需要创建自己可写的公共仓库,也不用将自己的更新推送到自己的服务器, 你只需将每次提交的差异内容以电子邮件的方式依次发送到邮件列表中即可。 $ git checkout -b topicA $ (work) $ git commit $ (work) $ git commit 如此一番后,有了两个提交要发到邮件列表。我们可以用 git format-patch 命令来生 成 mbox 格式的文件然后作为附件发送。每个提交都会封装为一个 .patch 后缀的 mbox 文 件,但其中只包含一封邮件,邮件标题就是提交消息(译注:额外有前缀,看例子),邮件 内容包含补丁正文和 Git 版本号。这种方式的妙处在于接受补丁时仍可保留原来的提交消 息,请看接下来的例子: $ git format-patch -M origin/master 0001-add-limit-to-log-function.patch 0002-changed-log-output-to-30-from-25.patch 112 Scott Chacon Pro Git 5.2节 为项目作贡献 format-patch 命令依次创建补丁文件,并输出文件名。上面的 -M 选项允许 Git 检查是 否有对文件重命名的提交。我们来看看补丁文件的内容: $ cat 0001-add-limit-to-log-function.patch From 330090432754092d704da8e76ca5c05c198e71a8 Mon Sep 17 00:00:00 2001 From: Jessica Smith Date: Sun, 6 Apr 2008 10:17:23 -0700 Subject: [PATCH 1/2] add limit to log function Limit log functionality to the first 20 --- lib/simplegit.rb | 2 +- 1 files changed, 1 insertions(+), 1 deletions(-) diff --git a/lib/simplegit.rb b/lib/simplegit.rb index 76f47bc..f9815f1 100644 --- a/lib/simplegit.rb +++ b/lib/simplegit.rb @@ -14,7 +14,7 @@ class SimpleGit end def log(treeish = 'master') - command("git log #{treeish}") + command("git log -n 20 #{treeish}") end def ls_tree(treeish = 'master') -- 1.6.2.rc1.20.g8c5b.dirty 如果有额外信息需要补充,但又不想放在提交消息中说明,可以编辑这些补丁文件, 在第一个 --- 行之前添加说明,但不要修改下面的补丁正文,比如例子中的 Limit log functionality to the first 20 部分。这样,其它开发者能阅读,但在采纳补丁时不会将 此合并进来。 你可以用邮件客户端软件发送这些补丁文件,也可以直接在命令行发送。有些所谓智能 的邮件客户端软件会自作主张帮你调整格式,所以粘贴补丁到邮件正文时,有可能会丢 失换行符和若干空格。Git 提供了一个通过 IMAP 发送补丁文件的工具。接下来我会演 示如何通过 Gmail 的 IMAP 服务器发送。另外,在 Git 源代码中有个 Documentation/ SubmittingPatches 文件,可以仔细读读,看看其它邮件程序的相关导引。 首先在 ~/.gitconfig 文件中配置 imap 项。每个选项都可用 git config 命令分别设 置,当然直接编辑文件添加以下内容更便捷: [imap] folder = "[Gmail]/Drafts" host = imaps://imap.gmail.com user = user@gmail.com pass = p4ssw0rd port = 993 sslverify = false 113 第5章 分布式 Git Scott Chacon Pro Git 如果你的 IMAP 服务器没有启用 SSL,就无需配置最后那两行,并且 host 应该以 imap:// 开头而不再是有 s 的 imaps://。保存配置文件后,就能用 git send-email 命令把补丁作 为邮件依次发送到指定的 IMAP 服务器上的文件夹中(译注:这里就是 Gmail 的 [Gmail]/ Drafts 文件夹。但如果你的语言设置不是英文,此处的文件夹 Drafts 字样会变为对应的 语言。): $ git send-email *.patch 0001-added-limit-to-log-function.patch 0002-changed-log-output-to-30-from-25.patch Who should the emails appear to be from? [Jessica Smith ] Emails will be sent from: Jessica Smith Who should the emails be sent to? jessica@example.com Message-ID to be used as In-Reply-To for the first email? y 接下来,Git 会根据每个补丁依次输出类似下面的日志: (mbox) Adding cc: Jessica Smith from \line 'From: Jessica Smith ' OK. Log says: Sendmail: /usr/sbin/sendmail -i jessica@example.com From: Jessica Smith To: jessica@example.com Subject: [PATCH 1/2] added limit to log function Date: Sat, 30 May 2009 13:29:15 -0700 Message-Id: <1243715356-61726-1-git-send-email-jessica@example.com> X-Mailer: git-send-email 1.6.2.rc1.20.g8c5b.dirty In-Reply-To: References: Result: OK 最后,到 Gmail 上打开 Drafts 文件夹,编辑这些邮件,修改收件人地址为邮件列表地 址,另外给要抄送的人也加到 Cc 列表中,最后发送。 5.2.6 小结 本节主要介绍了常见 Git 项目协作的工作流程,还有一些帮助处理这些工作的命令和工 具。接下来我们要看看如何维护 Git 项目,并成为一个合格的项目管理员,或是集成经 理。 5.3 项目的管理 既然是相互协作,在贡献代码的同时,也免不了要维护管理自己的项目。像是怎么处理别 人用 format-patch 生成的补丁,或是集成远端仓库上某个分支上的变化等等。但无论是管 理代码仓库,还是帮忙审核收到的补丁,都需要同贡献者约定某种长期可持续的工作方式。 114 Scott Chacon Pro Git 5.3节 项目的管理 5.3.1 使用特性分支进行工作 如果想要集成新的代码进来,最好局限在特性分支上做。临时的特性分支可以让你随意尝 试,进退自如。比如碰上无法正常工作的补丁,可以先搁在那边,直到有时间仔细核查修复 为止。创建的分支可以用相关的主题关键字命名,比如 ruby_client 或者其它类似的描述 性词语,以帮助将来回忆。Git 项目本身还时常把分支名称分置于不同命名空间下,比如 sc/ruby_client 就说明这是 sc 这个人贡献的。现在从当前主干分支为基础,新建临时分 支: $ git branch sc/ruby_client master 另外,如果你希望立即转到分支上去工作,可以用 checkout -b: $ git checkout -b sc/ruby_client master 好了,现在已经准备妥当,可以试着将别人贡献的代码合并进来了。之后评估一下有没有 问题,最后再决定是不是真的要并入主干。 5.3.2 采纳来自邮件的补丁 如果收到一个通过电邮发来的补丁,你应该先把它应用到特性分支上进行评估。有两种应 用补丁的方法:git apply 或者 git am。 使用 apply 命令应用补丁 如果收到的补丁文件是用 git diff 或由其它 Unix 的 diff 命令生成,就该用 git apply 命令来应用补丁。假设补丁文件存在 /tmp/patch-ruby-client.patch,可以这样运 行: $ git apply /tmp/patch-ruby-client.patch 这会修改当前工作目录下的文件,效果基本与运行 patch -p1 打补丁一样,但它更为 严格,且不会出现混乱。如果是 git diff 格式描述的补丁,此命令还会相应地添加,删 除,重命名文件。当然,普通的 patch 命令是不会这么做的。另外请注意,git apply 是 一个事务性操作的命令,也就是说,要么所有补丁都打上去,要么全部放弃。所以不会出现 patch 命令那样,一部分文件打上了补丁而另一部分却没有,这样一种不上不下的修订状 态。所以总的来说,git apply 要比 patch 严谨许多。因为仅仅是更新当前的文件,所以 此命令不会自动生成提交对象,你得手工缓存相应文件的更新状态并执行提交命令。 在实际打补丁之前,可以先用 git apply --check 查看补丁是否能够干净顺利地应用到 当前分支中: $ git apply --check 0001-seeing-if-this-helps-the-gem.patch error: patch failed: ticgit.gemspec:1 error: ticgit.gemspec: patch does not apply 115 第5章 分布式 Git Scott Chacon Pro Git 如果没有任何输出,表示我们可以顺利采纳该补丁。如果有问题,除了报告错误信息之 外,该命令还会返回一个非零的状态,所以在 shell 脚本里可用于检测状态。 使用 am 命令应用补丁 如果贡献者也用 Git,且擅于制作 format-patch 补丁,那你的合并工作将会非常轻松。 因为这些补丁中除了文件内容差异外,还包含了作者信息和提交消息。所以请鼓励贡献者用 format-patch 生成补丁。对于传统的 diff 命令生成的补丁,则只能用 git apply 处理。 对于 format-patch 制作的新式补丁,应当使用 git am 命令。从技术上来说,git am 能 够读取 mbox 格式的文件。这是种简单的纯文本文件,可以包含多封电邮,格式上用 From 加空格以及随便什么辅助信息所组成的行作为分隔行,以区分每封邮件,就像这样: From 330090432754092d704da8e76ca5c05c198e71a8 Mon Sep 17 00:00:00 2001 From: Jessica Smith Date: Sun, 6 Apr 2008 10:17:23 -0700 Subject: [PATCH 1/2] add limit to log function Limit log functionality to the first 20 这是 format-patch 命令输出的开头几行,也是一个有效的 mbox 文件格式。如果有人用 git send-email 给你发了一个补丁,你可以将此邮件下载到本地,然后运行 git am 命令 来应用这个补丁。如果你的邮件客户端能将多封电邮导出为 mbox 格式的文件,就可以用 git am 一次性应用所有导出的补丁。 如果贡献者将 format-patch 生成的补丁文件上传到类似 Request Ticket 一样的任务处 理系统,那么可以先下载到本地,继而使用 git am 应用该补丁: $ git am 0001-limit-log-function.patch Applying: add limit to log function 你会看到它被干净地应用到本地分支,并自动创建了新的提交对象。作者信息取自邮件头 From 和 Date,提交消息则取自 Subject 以及正文中补丁之前的内容。来看具体实例,采 纳之前展示的那个 mbox 电邮补丁后,最新的提交对象为: $ git log --pretty=fuller -1 commit 6c5e70b984a60b3cecd395edd5b48a7575bf58e0 Author: Jessica Smith AuthorDate: Sun Apr 6 10:17:23 2008 -0700 Commit: Scott Chacon CommitDate: Thu Apr 9 09:19:06 2009 -0700 add limit to log function Limit log functionality to the first 20 Commit 部分显示的是采纳补丁的人,以及采纳的时间。而 Author 部分则显示的是原作 者,以及创建补丁的时间。 116 Scott Chacon Pro Git 5.3节 项目的管理 有时,我们也会遇到打不上补丁的情况。这多半是因为主干分支和补丁的基础分支相差 太远,但也可能是因为某些依赖补丁还未应用。这种情况下,git am 会报错并询问该怎么 做: $ git am 0001-seeing-if-this-helps-the-gem.patch Applying: seeing if this helps the gem error: patch failed: ticgit.gemspec:1 error: ticgit.gemspec: patch does not apply Patch failed at 0001. When you have resolved this problem run "git am --resolved". If you would prefer to skip this patch, instead run "git am --skip". To restore the original branch and stop patching run "git am --abort". Git 会在有冲突的文件里加入冲突解决标记,这同合并或衍合操作一样。解决的办法也一 样,先编辑文件消除冲突,然后暂存文件,最后运行 git am --resolved 提交修正结果: $ (fix the file) $ git add ticgit.gemspec $ git am --resolved Applying: seeing if this helps the gem 如果想让 Git 更智能地处理冲突,可以用 -3 选项进行三方合并。如果当前分支未包含 该补丁的基础代码或其祖先,那么三方合并就会失败,所以该选项默认为关闭状态。一般来 说,如果该补丁是基于某个公开的提交制作而成的话,你总是可以通过同步来获取这个共同 祖先,所以用三方合并选项可以解决很多麻烦: $ git am -3 0001-seeing-if-this-helps-the-gem.patch Applying: seeing if this helps the gem error: patch failed: ticgit.gemspec:1 error: ticgit.gemspec: patch does not apply Using index info to reconstruct a base tree... Falling back to patching base and 3-way merge... No changes -- Patch already applied. 像上面的例子,对于打过的补丁我又再打一遍,自然会产生冲突,但因为加上了 -3 选 项,所以它很聪明地告诉我,无需更新,原有的补丁已经应用。 对于一次应用多个补丁时所用的 mbox 格式文件,可以用 am 命令的交互模式选项 -i, 这样就会在打每个补丁前停住,询问该如何操作: $ git am -3 -i mbox Commit Body is: -------------------------- seeing if this helps the gem -------------------------- Apply? [y]es/[n]o/[e]dit/[v]iew patch/[a]ccept all 117 第5章 分布式 Git Scott Chacon Pro Git 在多个补丁要打的情况下,这是个非常好的办法,一方面可以预览下补丁内容,同时也可 以有选择性的接纳或跳过某些补丁。 打完所有补丁后,如果测试下来新特性可以正常工作,那就可以安心地将当前特性分支合 并到长期分支中去了。 5.3.3 检出远程分支 如果贡献者有自己的 Git 仓库,并将修改推送到此仓库中,那么当你拿到仓库的访问地 址和对应分支的名称后,就可以加为远程分支,然后在本地进行合并。 比如,Jessica 发来一封邮件,说在她代码库中的 ruby-client 分支上已经实现了某个 非常棒的新功能,希望我们能帮忙测试一下。我们可以先把她的仓库加为远程仓库,然后抓 取数据,完了再将她所说的分支检出到本地来测试: $ git remote add jessica git://github.com/jessica/myproject.git $ git fetch jessica $ git checkout -b rubyclient jessica/ruby-client 若是不久她又发来邮件,说还有个很棒的功能实现在另一分支上,那我们只需重新抓取下 最新数据,然后检出那个分支到本地就可以了,无需重复设置远程仓库。 这种做法便于同别人保持长期的合作关系。但前提是要求贡献者有自己的服务器,而我们 也需要为每个人建一个远程分支。有些贡献者提交代码补丁并不是很频繁,所以通过邮件接 收补丁效率会更高。同时我们自己也不会希望建上百来个分支,却只从每个分支取一两个补 丁。但若是用脚本程序来管理,或直接使用代码仓库托管服务,就可以简化此过程。当然, 选择何种方式取决于你和贡献者的喜好。 使用远程分支的另外一个好处是能够得到提交历史。不管代码合并是不是会有问题,至少 我们知道该分支的历史分叉点,所以默认会从共同祖先开始自动进行三方合并,无需 -3 选 项,也不用像打补丁那样祈祷存在共同的基准点。 如果只是临时合作,只需用 git pull 命令抓取远程仓库上的数据,合并到本地临时分支 就可以了。一次性的抓取动作自然不会把该仓库地址加为远程仓库。 $ git pull git://github.com/onetimeguy/project.git From git://github.com/onetimeguy/project * branch HEAD -> FETCH_HEAD Merge made by recursive. 5.3.4 决断代码取舍 现在特性分支上已合并好了贡献者的代码,是时候决断取舍了。本节将回顾一些之前学过 的命令,以看清将要合并到主干的是哪些代码,从而理解它们到底做了些什么,是否真的要 并入。 一般我们会先看下,特性分支上都有哪些新增的提交。比如在 contrib 特性分支上打了 两个补丁,仅查看这两个补丁的提交信息,可以用 --not 选项指定要屏蔽的分支 master, 这样就会剔除重复的提交历史: 118 Scott Chacon Pro Git 5.3节 项目的管理 $ git log contrib --not master commit 5b6235bd297351589efc4d73316f0a68d484f118 Author: Scott Chacon Date: Fri Oct 24 09:53:59 2008 -0700 seeing if this helps the gem commit 7482e0d16d04bea79d0dba8988cc78df655f16a0 Author: Scott Chacon Date: Mon Oct 22 19:38:36 2008 -0700 updated the gemspec to hopefully work better 还可以查看每次提交的具体修改。请牢记,在 git log 后加 -p 选项将展示每次提交的 内容差异。 如果想看当前分支同其他分支合并时的完整内容差异,有个小窍门: $ git diff master 虽然能得到差异内容,但请记住,结果有可能和我们的预期不同。一旦主干 master 在特 性分支创建之后有所修改,那么通过 diff 命令来比较的,是最新主干上的提交快照。显 然,这不是我们所要的。比方在 master 分支中某个文件里添了一行,然后运行上面的命 令,简单的比较最新快照所得到的结论只能是,特性分支中删除了这一行。 这个很好理解:如果 master 是特性分支的直接祖先,不会产生任何问题;如果它们的提 交历史在不同的分叉上,那么产生的内容差异,看起来就像是增加了特性分支上的新代码, 同时删除了 master 分支上的新代码。 实际上我们真正想要看的,是新加入到特性分支的代码,也就是合并时会并入主干的代 码。所以,准确地讲,我们应该比较特性分支和它同 master 分支的共同祖先之间的差异。 我们可以手工定位它们的共同祖先,然后与之比较: $ git merge-base contrib master 36c7dba2c95e6bbb78dfa822519ecfec6e1ca649 $ git diff 36c7db 但这么做很麻烦,所以 Git 提供了便捷的 ... 语法。对于 diff 命令,可以把 ... 加 在原始分支(拥有共同祖先)和当前分支之间: $ git diff master...contrib 现在看到的,就是实际将要引入的新代码。这是一个非常有用的命令,应该牢记。 5.3.5 代码集成 一俟特性分支准备停当,接下来的问题就是如何集成到更靠近主线的分支中。此外还要考 虑维护项目的总体步骤是什么。虽然有很多选择,不过我们这里只介绍其中一部分。 119 第5章 分布式 Git Scott Chacon Pro Git 合并流程 一般最简单的情形,是在 master 分支中维护稳定代码,然后在特性分支上开发新功能, 或是审核测试别人贡献的代码,接着将它并入主干,最后删除这个特性分支,如此反复。来 看示例,假设当前代码库中有两个分支,分别为 ruby_client 和 php_client,如图 5.19 所示。然后先把 ruby_client 合并进主干,再合并 php_client,最后的提交历史如图 5.20 所示。 图 5.19: 多个特性分支 图 5.20: 合并特性分支之后 这是最简单的流程,所以在处理大一些的项目时可能会有问题。 对于大型项目,至少需要维护两个长期分支 master 和 develop。新代码(图 5.21 中 的 ruby_client)将首先并入 develop 分支(图 5.22 中的 C8),经过一个阶段,确认 develop 中的代码已稳定到可发行时,再将 master 分支快进到稳定点(图 5.23 中的 C8)。而平时这两个分支都会被推送到公开的代码库。 120 Scott Chacon Pro Git 5.3节 项目的管理 图 5.21: 特性分支合并前 图 5.22: 特性分支合并后 图 5.23: 特性分支发布后 这样,在人们克隆仓库时就有两种选择:既可检出最新稳定版本,确保正常使用;也能检 出开发版本,试用最前沿的新特性。 你也可以扩展这个概念,先将所有新代码合并到临时特性分支,等到该分支稳定下来并通 过测试后,再并入 develop 分支。然后,让时间检验一切,如果这些代码确实可以正常工 作相当长一段时间,那就有理由相信它已经足够稳定,可以放心并入主干分支发布。 121 第5章 分布式 Git Scott Chacon Pro Git 大项目的合并流程 Git 项目本身有四个长期分支:用于发布的 master 分支、用于合并基本稳定特性的 next 分支、用于合并仍需改进特性的 pu 分支(pu 是 proposed updates 的缩写),以及用于 除错维护的 maint 分支(maint 取自 maintenance)。维护者可以按照之前介绍的方法, 将贡献者的代码引入为不同的特性分支(如图 5.24 所示),然后测试评估,看哪些特性能 稳定工作,哪些还需改进。稳定的特性可以并入 next 分支,然后再推送到公共仓库,以供 其他人试用。 图 5.24: 管理复杂的并行贡献 仍需改进的特性可以先并入 pu 分支。直到它们完全稳定后再并入 master。同时一并检 查下 next 分支,将足够稳定的特性也并入 master。所以一般来说,master 始终是在快 进,next 偶尔做下衍合,而 pu 则是频繁衍合,如图 5.25 所示: 图 5.25: 将特性并入长期分支 并入 master 后的特性分支,已经无需保留分支索引,放心删除好了。Git 项目还有一个 maint 分支,它是以最近一次发行版为基础分化而来的,用于维护除错补丁。所以克隆 Git 项目仓库后会得到这四个分支,通过检出不同分支可以了解各自进展,或是试用前沿特性, 或是贡献代码。而维护者则通过管理这些分支,逐步有序地并入第三方贡献。 衍合与挑拣(cherry-pick)的流程 一些维护者更喜欢衍合或者挑拣贡献者的代码,而不是简单的合并,因为这样能够保持线 性的提交历史。如果你完成了一个特性的开发,并决定将它引入到主干代码中,你可以转到 那个特性分支然后执行衍合命令,好在你的主干分支上(也可能是develop分支之类的)重 122 Scott Chacon Pro Git 5.3节 项目的管理 新提交这些修改。如果这些代码工作得很好,你就可以快进master分支,得到一个线性的提 交历史。 另一个引入代码的方法是挑拣。挑拣类似于针对某次特定提交的衍合。它首先提取某次提 交的补丁,然后试着应用在当前分支上。如果某个特性分支上有多个commits,但你只想引 入其中之一就可以使用这种方法。也可能仅仅是因为你喜欢用挑拣,讨厌衍合。假设你有一 个类似图 5.26的工程。 图 5.26: 挑拣(cherry-pick)之前的历史 如果你希望拉取e43a6到你的主干分支,可以这样: $ git cherry-pick e43a6fd3e94888d76779ad79fb568ed180e5fcdf Finished one cherry-pick. [master]: created a0a41a9: "More friendly message when locking the index fails." 3 files changed, 17 insertions(+), 3 deletions(-) 这将会引入e43a6的代码,但是会得到不同的SHA-1值,因为应用日期不同。现在你的历史 看起来像图 5.27. 图 5.27: 挑拣(cherry-pick)之后的历史 现在,你可以删除这个特性分支并丢弃你不想引入的那些commit。 5.3.6 给发行版签名 你可以删除上次发布的版本并重新打标签,也可以像第二章所说的那样建立一个新的标 签。如果你决定以维护者的身份给发行版签名,应该这样做: 123 第5章 分布式 Git Scott Chacon Pro Git $ git tag -s v1.5 -m 'my signed 1.5 tag' You need a passphrase to unlock the secret key for user: "Scott Chacon " 1024-bit DSA key, ID F721C45A, created 2009-02-09 完成签名之后,如何分发PGP公钥(public key)是个问题。(译者注:分发公钥是为了 验证标签)。还好,Git的设计者想到了解决办法:可以把key(既公钥)作为blob变量写 入Git库,然后把它的内容直接写在标签里。gpg --list-keys命令可以显示出你所拥有的 key: $ gpg --list-keys /Users/schacon/.gnupg/pubring.gpg --------------------------------- pub 1024D/F721C45A 2009-02-09 [expires: 2010-02-09] uid Scott Chacon sub 2048g/45D02282 2009-02-09 [expires: 2010-02-09] 然后,导出key的内容并经由管道符传递给git hash-object,之后钥匙会以blob类型写入 Git中,最后返回这个blob量的SHA-1值: $ gpg -a --export F721C45A | git hash-object -w --stdin 659ef797d181633c87ec71ac3f9ba29fe5775b92 现在你的Git已经包含了这个key的内容了,可以通过不同的SHA-1值指定不同的key来创建 标签。 $ git tag -a maintainer-pgp-pub 659ef797d181633c87ec71ac3f9ba29fe5775b92 在运行git push --tags命令之后,maintainer-pgp-pub标签就会公布给所有人。如果有 人想要校验标签,他可以使用如下命令导入你的key: $ git show maintainer-pgp-pub | gpg --import 人们可以用这个key校验你签名的所有标签。另外,你也可以在标签信息里写入一个操作 向导,用户只需要运行git show 查看标签信息,然后按照你的向导就能完成校验。 5.3.7 生成内部版本号 因为Git不会为每次提交自动附加类似’v123’的递增序列,所以如果你想要得到一个便 于理解的提交号可以运行git describe命令。Git将会返回一个字符串,由三部分组成:最 近一次标定的版本号,加上自那次标定之后的提交次数,再加上一段SHA-1值of the commit you’re describing: 124 Scott Chacon Pro Git 5.3节 项目的管理 $ git describe master v1.6.2-rc1-20-g8c5b85c 这个字符串可以作为快照的名字,方便人们理解。如果你的Git是你自己下载源码然后编 译安装的,你会发现git --version命令的输出和这个字符串差不多。如果在一个刚刚打完 标签的提交上运行describe命令,只会得到这次标定的版本号,而没有后面两项信息。 git describe命令只适用于有标注的标签(通过-a或者-s选项创建的标签),所以发行版 的标签都应该是带有标注的,以保证git describe能够正确的执行。你也可以把这个字符串 作为checkout或者show命令的目标,因为他们最终都依赖于一个简短的SHA-1值,当然如果 这个SHA-1值失效他们也跟着失效。最近Linux内核为了保证SHA-1值的唯一性,将位数由8位 扩展到10位,这就导致扩展之前的git describe输出完全失效了。 5.3.8 准备发布 现在可以发布一个新的版本了。首先要将代码的压缩包归档,方便那些可怜的还没有使用 Git的人们。可以使用git archive: $ git archive master --prefix='project/' | gzip > `git describe master`.tar.gz $ ls *.tar.gz v1.6.2-rc1-20-g8c5b85c.tar.gz 这个压缩包解压出来的是一个文件夹,里面是你项目的最新代码快照。你也可以用类似的 方法建立一个zip压缩包,在git archive加上--format=zip选项: $ git archive master --prefix='project/' --format=zip > `git describe master`.zip 现在你有了一个tar.gz压缩包和一个zip压缩包,可以把他们上传到你网站上或者用 e-mail发给别人。 5.3.9 制作简报 是时候通知邮件列表里的朋友们来检验你的成果了。使用git shortlog命令可以方便快 捷的制作一份修改日志(changelog),告诉大家上次发布之后又增加了哪些特性和修复 了哪些bug。实际上这个命令能够统计给定范围内的所有提交;假如你上一次发布的版本是 v1.0.1,下面的命令将给出自从上次发布之后的所有提交的简介: $ git shortlog --no-merges master --not v1.0.1 Chris Wanstrath (8): Add support for annotated tags to Grit::Tag Add packed-refs annotated tag support. Add Grit::Commit#to_patch Update version and History.txt Remove stray `puts` 125 第5章 分布式 Git Scott Chacon Pro Git Make ls_tree ignore nils Tom Preston-Werner (4): fix dates in history dynamic version method Version bump to 1.0.2 Regenerated gemspec for version 1.0.2 这就是自从v1.0.1版本以来的所有提交的简介,内容按照作者分组,以便你能快速的发 e-mail给他们。 5.4 小结 你学会了如何使用Git为项目做贡献,也学会了如何使用Git维护你的项目。恭喜!你已经 成为一名高效的开发者。在下一章你将学到更强大的工具来处理更加复杂的问题,之后你会 变成一位Git大师。 126 第6章 Git 工具 现在,你已经学习了管理或者维护 Git 仓库,实现代码控制所需的大多数日常命令和工 作流程。你已经完成了跟踪和提交文件的基本任务,并且发挥了暂存区和轻量级的特性分支 及合并的威力。 接下来你将领略到一些 Git 可以实现的非常强大的功能,这些功能你可能并不会在日常 操作中使用,但在某些时候你也许会需要。 6.1 修订版本(Revision)选择 Git 允许你通过几种方法来指明特定的或者一定范围内的提交。了解它们并不是必需的, 但是了解一下总没坏处。 6.1.1 单个修订版本 显然你可以使用给出的 SHA-1 值来指明一次提交,不过也有更加人性化的方法来做同样 的事。本节概述了指明单个提交的诸多方法。 6.1.2 简短的SHA Git 很聪明,它能够通过你提供的前几个字符来识别你想要的那次提交,只要你提供的那 部分 SHA-1 不短于四个字符,并且没有歧义——也就是说,当前仓库中只有一个对象以这 段 SHA-1 开头。 例如,想要查看一次指定的提交,假设你运行 git log 命令并找到你增加了功能的那次 提交: $ git log commit 734713bc047d87bf7eac9674765ae793478c50d3 Author: Scott Chacon Date: Fri Jan 2 18:32:33 2009 -0800 fixed refs handling, added gc auto, updated tests commit d921970aadf03b3cf0e71becdaab3147ba71cdef Merge: 1c002dd... 35cfb2b... Author: Scott Chacon 127 第6章 Git 工具 Scott Chacon Pro Git Date: Thu Dec 11 15:08:43 2008 -0800 Merge commit 'phedders/rdocs' commit 1c002dd4b536e7479fe34593e72e6c6c1819e53b Author: Scott Chacon Date: Thu Dec 11 14:58:32 2008 -0800 added some blame and merge stuff 假设是 1c002dd.... 。如果你想 git show 这次提交,下面的命令是等价的(假设简短 的版本没有歧义): $ git show 1c002dd4b536e7479fe34593e72e6c6c1819e53b $ git show 1c002dd4b536e7479f $ git show 1c002d Git 可以为你的 SHA-1 值生成出简短且唯一的缩写。如果你传递 --abbrev-commit 给 git log 命令,输出结果里就会使用简短且唯一的值;它默认使用七个字符来表示,不过必 要时为了避免 SHA-1 的歧义,会增加字符数: $ git log --abbrev-commit --pretty=oneline ca82a6d changed the verison number 085bb3b removed unnecessary test code a11bef0 first commit 通常在一个项目中,使用八到十个字符来避免 SHA-1 歧义已经足够了。最大的 Git 项目 之一,Linux 内核,目前也只需要最长 40 个字符中的 12 个字符来保持唯一性。 6.1.3 关于 SHA-1 的简短说明 许多人可能会担心一个问题:在随机的偶然情况下,在他们的仓库里会出现两个具有相同 SHA-1 值的对象。那会怎么样呢? 如果你真的向仓库里提交了一个跟之前的某个对象具有相同 SHA-1 值的对象,Git 将会 发现之前的那个对象已经存在在 Git 数据库中,并认为它已经被写入了。如果什么时候你 想再次检出那个对象时,你会总是得到先前的那个对象的数据。 不过,你应该了解到,这种情况发生的概率是多么微小。SHA-1 摘要长度是 20 字节,也 就是 160 位。为了保证有 50% 的概率出现一次冲突,需要 280 个随机哈希的对象(计算 冲突机率的公式是 p = n(n−1) 2 × 1 2160 )。280 是 1.2×1024,也就是一亿亿亿,那是地球上沙 粒总数的 1200 倍。 现在举例说一下怎样才能产生一次 SHA-1 冲突。如果地球上 65 亿的人类都在编程,每 人每秒都在产生等价于整个 Linux 内核历史(一百万个 Git 对象)的代码,并将之提交到 一个巨大的 Git 仓库里面,那将花费 5 年的时间才会产生足够的对象,使其拥有 50% 的 概率产生一次 SHA-1 对象冲突。这要比你编程团队的成员同一个晚上在互不相干的意外中 被狼袭击并杀死的机率还要小。 128 Scott Chacon Pro Git 6.1节 修订版本(Revision)选择 6.1.4 分支引用 指明一次提交的最直接的方法要求有一个指向它的分支引用。这样,你就可以在任何需要 一个提交对象或者 SHA-1 值的 Git 命令中使用该分支名称了。如果你想要显示一个分支的 最后一次提交的对象,例如假设 topic1 分支指向 ca82a6d,那么下面的命令是等价的: $ git show ca82a6dff817ec66f44342007202690a93763949 $ git show topic1 如果你想知道某个分支指向哪个特定的 SHA,或者想看任何一个例子中被简写的 SHA-1, 你可以使用一个叫做 rev-parse 的 Git 探测工具。在第9章 你可以看到关于探测工具 的更多信息;简单来说,rev-parse 是为了底层操作而不是日常操作设计的。不过,有时 你想看 Git 现在到底处于什么状态时,它可能会很有用。这里你可以对你的分支运执行 rev-parse。 $ git rev-parse topic1 ca82a6dff817ec66f44342007202690a93763949 6.1.5 引用日志里的简称 在你工作的同时,Git 在后台的工作之一就是保存一份引用日志——一份记录最近几个月 你的 HEAD 和分支引用的日志。 你可以使用 git reflog 来查看引用日志: $ git reflog 734713b... HEAD@{0}: commit: fixed refs handling, added gc auto, updated d921970... HEAD@{1}: merge phedders/rdocs: Merge made by recursive. 1c002dd... HEAD@{2}: commit: added some blame and merge stuff 1c36188... HEAD@{3}: rebase -i (squash): updating HEAD 95df984... HEAD@{4}: commit: # This is a combination of two commits. 1c36188... HEAD@{5}: rebase -i (squash): updating HEAD 7e05da5... HEAD@{6}: rebase -i (pick): updating HEAD 每次你的分支顶端因为某些原因被修改时,Git 就会为你将信息保存在这个临时历史记录 里面。你也可以使用这份数据来指明更早的分支。如果你想查看仓库中 HEAD 在五次前的 值,你可以使用引用日志的输出中的 @n 引用: $ git show HEAD@{5} 你也可以使用这个语法来查看一定时间前分支指向哪里。例如,想看你的 master 分支昨 天在哪,你可以输入 129 第6章 Git 工具 Scott Chacon Pro Git $ git show master@{yesterday} 它就会显示昨天分支的顶端在哪。这项技术只对还在你引用日志里的数据有用,所以不能 用来查看比几个月前还早的提交。 想要看类似于 git log 输出格式的引用日志信息,你可以运行 git log -g: $ git log -g master commit 734713bc047d87bf7eac9674765ae793478c50d3 Reflog: master@{0} (Scott Chacon ) Reflog message: commit: fixed refs handling, added gc auto, updated Author: Scott Chacon Date: Fri Jan 2 18:32:33 2009 -0800 fixed refs handling, added gc auto, updated tests commit d921970aadf03b3cf0e71becdaab3147ba71cdef Reflog: master@{1} (Scott Chacon ) Reflog message: merge phedders/rdocs: Merge made by recursive. Author: Scott Chacon Date: Thu Dec 11 15:08:43 2008 -0800 Merge commit 'phedders/rdocs' 需要注意的是,日志引用信息只存在于本地——这是一个你在仓库里做过什么的日志。其 他人的仓库拷贝里的引用和你的相同;而你新克隆一个仓库的时候,引用日志是空的,因为 你在仓库里还没有操作。只有你克隆了一个项目至少两个月,git show HEAD@2.months.ago 才会有用——如果你是五分钟前克隆的仓库,将不会有结果返回。 6.1.6 祖先引用 另一种指明某次提交的常用方法是通过它的祖先。如果你在引用最后加上一个 ˆ,Git 将 其理解为此次提交的父提交。 假设你的工程历史是这样的: $ git log --pretty=format:'%h %s' --graph * 734713b fixed refs handling, added gc auto, updated tests * d921970 Merge commit 'phedders/rdocs' |\ | * 35cfb2b Some rdoc changes * | 1c002dd added some blame and merge stuff |/ * 1c36188 ignore *.gem * 9b29157 add open3_detach to gemspec file list 那么,想看上一次提交,你可以使用 HEADˆ,意思是“HEAD 的父提交”: 130 Scott Chacon Pro Git 6.1节 修订版本(Revision)选择 $ git show HEAD^ commit d921970aadf03b3cf0e71becdaab3147ba71cdef Merge: 1c002dd... 35cfb2b... Author: Scott Chacon Date: Thu Dec 11 15:08:43 2008 -0800 Merge commit 'phedders/rdocs' 你也可以在 ˆ 后添加一个数字——例如,d921970ˆ2 意思是“d921970 的第二父提交”。 这种语法只在合并提交时有用,因为合并提交可能有多个父提交。第一父提交是你合并时所 在分支,而第二父提交是你所合并的分支: $ git show d921970^ commit 1c002dd4b536e7479fe34593e72e6c6c1819e53b Author: Scott Chacon Date: Thu Dec 11 14:58:32 2008 -0800 added some blame and merge stuff $ git show d921970^2 commit 35cfb2b795a55793d7cc56a6cc2060b4bb732548 Author: Paul Hedderly Date: Wed Dec 10 22:22:03 2008 +0000 Some rdoc changes 另外一个指明祖先提交的方法是 ~。这也是指向第一父提交,所以 HEAD~ 和 HEADˆ 是等 价的。当你指定数字的时候就明显不一样了。HEAD~2 是指“第一父提交的第一父提交”, 也就是“祖父提交”——它会根据你指定的次数检索第一父提交。例如,在上面列出的历史 记录里面,HEAD~3 会是 $ git show HEAD~3 commit 1c3618887afb5fbcbea25b7c013f4e2114448b8d Author: Tom Preston-Werner Date: Fri Nov 7 13:47:59 2008 -0500 ignore *.gem 也可以写成 HEADˆˆˆ,同样是第一父提交的第一父提交的第一父提交: $ git show HEAD^^^ commit 1c3618887afb5fbcbea25b7c013f4e2114448b8d Author: Tom Preston-Werner Date: Fri Nov 7 13:47:59 2008 -0500 ignore *.gem 131 第6章 Git 工具 Scott Chacon Pro Git 你也可以混合使用这些语法——你可以通过 HEAD~3ˆ2 指明先前引用的第二父提交(假设 它是一个合并提交)。 6.1.7 提交范围 现在你已经可以指明单次的提交,让我们来看看怎样指明一定范围的提交。这在你管理 分支的时候尤显重要——如果你有很多分支,你可以指明范围来圈定一些问题的答案,比 如:“这个分支上我有哪些工作还没合并到主分支的?” 双点 最常用的指明范围的方法是双点的语法。这种语法主要是让 Git 区分出可从一个分支中 获得而不能从另一个分支中获得的提交。例如,假设你有类似于图 6.1 的提交历史。 图 6.1: 范围选择的提交历史实例 你想要查看你的试验分支上哪些没有被提交到主分支,那么你就可以使用 master..experiment 来让 Git 显示这些提交的日志——这句话的意思是“所有可从experiment分支中获得而不 能从master分支中获得的提交”。为了使例子简单明了,我使用了图标中提交对象的字母来 代替真实日志的输出,所以会显示: $ git log master..experiemnt D C 另一方面,如果你想看相反的——所有在 master 而不在 experiment 中的分支——你可 以交换分支的名字。experiment..master 显示所有可在 master 获得而在 experiment 中 不能的提交: $ git log experiment..master F E 这在你想保持 experiment 分支最新和预览你将合并的提交的时候特别有用。这个语法的 另一种常见用途是查看你将把什么推送到远程: $ git log origin/master..HEAD 这条命令显示任何在你当前分支上而不在远程origin 上的提交。如果你运行 git push 并且的你的当前分支正在跟踪 origin/master,被git log origin/master..HEAD 列出的 132 Scott Chacon Pro Git 6.1节 修订版本(Revision)选择 提交就是将被传输到服务器上的提交。 你也可以留空语法中的一边来让 Git 来假定它是 HEAD。例如,输入 git log origin/master.. 将得到和上面的例子一样的结果—— Git 使 用 HEAD 来代替不存在的一边。 多点 双点语法就像速记一样有用;但是你也许会想针对两个以上的分支来指明修订版本,比如 查看哪些提交被包含在某些分支中的一个,但是不在你当前的分支上。Git允许你在引用前 使用ˆ字符或者--not指明你不希望提交被包含其中的分支。因此下面三个命令是等同的: $ git log refA..refB $ git log ^refA refB $ git log refB --not refA 这样很好,因为它允许你在查询中指定多于两个的引用,而这是双点语法所做不到的。例 如,如果你想查找所有从refA或refB包含的但是不被refC包含的提交,你可以输入下面中的 一个 $ git log refA refB ^refC $ git log refA refB --not refC 这建立了一个非常强大的修订版本查询系统,应该可以帮助你解决分支里包含了什么这个 问题。 三点 最后一种主要的范围选择语法是三点语法,这个可以指定被两个引用中的一个包含但又 不被两者同时包含的分支。回过头来看一下图6-1里所列的提交历史的例子。 如果你想查 看master或者experiment中包含的但不是两者共有的引用,你可以运行 $ git log master...experiment F E D C 这个再次给出你普通的log输出但是只显示那四次提交的信息,按照传统的提交日期排 列。 这种情形下,log命令的一个常用参数是--left-right,它会显示每个提交到底处于哪一 侧的分支。这使得数据更加有用。 $ git log --left-right master...experiment < F < E 133 第6章 Git 工具 Scott Chacon Pro Git > D > C 有了以上工具,让Git知道你要察看哪些提交就容易得多了。 6.2 交互式暂存 Git提供了很多脚本来辅助某些命令行任务。这里,你将看到一些交互式命令,它们帮助 你方便地构建只包含特定组合和部分文件的提交。在你修改了一大批文件然后决定将这些变 更分布在几个各有侧重的提交而不是单个又大又乱的提交时,这些工具非常有用。用这种方 法,你可以确保你的提交在逻辑上划分为相应的变更集,以便于供和你一起工作的开发者审 阅。 如果你运行git add时加上-i或者--interactive选项,Git就进入了一个交互式的shell模 式,显示一些类似于下面的信息: $ git add -i staged unstaged path 1: unchanged +0/-1 TODO 2: unchanged +1/-1 index.html 3: unchanged +5/-1 lib/simplegit.rb *** Commands *** 1: status 2: update 3: revert 4: add untracked 5: patch 6: diff 7: quit 8: help What now> 你会看到这个命令以一个完全不同的视图显示了你的暂存区——主要是你通过git sta- tus得到的那些信息但是稍微简洁但信息更加丰富一些。它在左侧列出了你暂存的变更,在 右侧列出了未被暂存的变更。 在这之后是一个命令区。这里你可以做很多事情,包括暂存文件,撤回文件,暂存部分文 件,加入未被追踪的文件,查看暂存文件的差别。 6.2.1 暂存和撤回文件 如果你在What now>的提示后输入2或者u,这个脚本会提示你那些文件你想要暂存: What now> 2 staged unstaged path 1: unchanged +0/-1 TODO 2: unchanged +1/-1 index.html 3: unchanged +5/-1 lib/simplegit.rb Update>> 如果想暂存TODO和index.html,你可以输入相应的编号: 134 Scott Chacon Pro Git 6.2节 交互式暂存 Update>> 1,2 staged unstaged path * 1: unchanged +0/-1 TODO * 2: unchanged +1/-1 index.html 3: unchanged +5/-1 lib/simplegit.rb Update>> 每个文件旁边的*表示选中的文件将被暂存。如果你在update>>提示后直接敲入回车,Git 会替你把所有选中的内容暂存: Update>> updated 2 paths *** Commands *** 1: status 2: update 3: revert 4: add untracked 5: patch 6: diff 7: quit 8: help What now> 1 staged unstaged path 1: +0/-1 nothing TODO 2: +1/-1 nothing index.html 3: unchanged +5/-1 lib/simplegit.rb 现在你可以看到TODO和index.html文件被暂存了同时simplegit.rb文件仍然未被暂存。如 果这时你想要撤回TODO文件,就使用3或者r(代表revert,恢复)选项: *** Commands *** 1: status 2: update 3: revert 4: add untracked 5: patch 6: diff 7: quit 8: help What now> 3 staged unstaged path 1: +0/-1 nothing TODO 2: +1/-1 nothing index.html 3: unchanged +5/-1 lib/simplegit.rb Revert>> 1 staged unstaged path * 1: +0/-1 nothing TODO 2: +1/-1 nothing index.html 3: unchanged +5/-1 lib/simplegit.rb Revert>> [enter] reverted one path 再次查看Git的状态,你会看到你已经撤回了TODO文件 *** Commands *** 1: status 2: update 3: revert 4: add untracked 5: patch 6: diff 7: quit 8: help What now> 1 135 第6章 Git 工具 Scott Chacon Pro Git staged unstaged path 1: unchanged +0/-1 TODO 2: +1/-1 nothing index.html 3: unchanged +5/-1 lib/simplegit.rb 要查看你暂存内容的差异,你可以使用6或者d(表示diff)命令。它会显示你暂存文件 的列表,你可以选择其中的几个,显示其被暂存的差异。这跟你在命令行下指定git diff --cached非常相似: *** Commands *** 1: status 2: update 3: revert 4: add untracked 5: patch 6: diff 7: quit 8: help What now> 6 staged unstaged path 1: +1/-1 nothing index.html Review diff>> 1 diff --git a/index.html b/index.html index 4d07108..4335f49 100644 --- a/index.html +++ b/index.html @@ -16,7 +16,7 @@ Date Finder

...

- +