Files
translations/git-workflows-and-tutorials/workflow-forking.md
2014-08-31 12:21:52 +08:00

8.8 KiB
Raw Blame History

Forking工作流

Forking工作流和前面讨论的几种工作流有根本的不同。 这种工作流不是使用单个服务端仓库作为『中央』代码基线,而让各个开发者都有一个服务端仓库。 这意味着各个代码贡献者不只有一个而2个Git仓库:一个本地私有的,另一个服务端公有的。

Forking工作流的一个主要优势是,贡献的代码可以被集成,而不需要所有人都push代码到仅有的中央仓库中。 开发者push到自己的服务端仓库,而只有项目维护者才能push到正式仓库。 这样项目维护者可以接受任何开发者的提交,但无需给他正式代码库的写权限。

效果就是一个发布式的工作流,为大型、自发性的团队(包括了不受信的第三方)提供了灵活的方式来安全的协作。 也让这个工作流成为开源项目的理想工作流。

🍺 工作方式

和其它的Git工作流一样,Forking工作流要先有一个公有的正式仓库存储在服务器上。 但一个新的开发者想要在项目上工作时,不是直接从正式仓库克隆。

而是fork正式项目在服务器上创建一个拷贝。这个仓库拷贝作为他个人公开仓库 —— 其它开发者不允许push到这个仓库,但可以pull到修改(后面我们很快就会看这点是很重要的)。 在创建了自己服务端拷贝之后,开发者执行git clone命令克隆仓库到本地机器上,作为私有的开发环境,和之前的工作流一样。

要提交本地修改时,push提交到自己公有仓库中 —— 而不是正式仓库中。 然后,给正式仓库发起一个pull request,让项目维护者知道有更新已经准备好可以集成了。 对于贡献的代码,pull request也可以很方便地作为一个讨论的地方。

为了集成功能到正式代码库,维护者pull贡献者的变更到自己的本地仓库中,检查变更以确保不会让项目出错, 合并变更到自己本地的master分支 然后push到服务器的master分支上。 到此,贡献的提交成为了项目的一部分,其它的开发者应该执行pull操作与正式仓库同步自己本地仓库。

正式仓库

Forking工作流中,『官方』仓库的叫法只是一个约定,理解这点很重要。 在技术观点上,各个开发者仓库和正式仓库在Git看来没有任何区别。 事实上,让正式仓库之所以正式是是看公开仓库的维护者是谁。

Forking工作流的分支使用方式

所有的个人分开仓库实际上只是为了方便和其它的开发者共享分支。各个开发者应该用分支隔离各个功能,就像在功能分支工作流Gitflow工作流一样。唯一的区别是这些分支被共享了,而功能分支工作流和Gitflow工作流是直接push到正式仓库中。

🍺 示例

项目维护者初始化正式仓库

和任何使用Git项目一样,第一步是创建在服务器上一个正式仓库,让团队成功可以访问。 通常这个仓库也会作为项目维护者的公有仓库。

公有仓库应该是裸仓库,不管是不是正式代码库。 所以项目维护者会运行你下面的命令来搭建正式仓库:

ssh user@host
git init --bare /path/to/repo.git

BitbucketStash提供了一个方便的GUI客户端以完成上面命令行做的事。 这个搭建中央仓库的过程和前面提到的工作流完全一样。 如果已有代码库,维护者也要push到这个仓库中。

开发者fork正式仓库

下一步,各个开发者会各自的公有仓库,用熟悉的git clone命令。

在这个示例中,假定用Bitbucket托管了仓库。记住,这样的话各个开发者需要各自的Bitbucket账号, 使用下面命令克隆服务端仓库:

git clone https://user@bitbucket.org/user/repo.git

相比前面介绍的工作流只用了一个origin远程别名指向中央仓库,Forking工作流需要2个远程别名 —— 一个指向正式仓库,另一个指向开发者自己的服务端仓库。别名的命名可以任意命名,常见的约定是使用origin作为远程克隆的仓库的别名 (这个别名会在运行git clone自动创建),upstream(上游)作为正式仓库的别名。

git remote add upstream https://bitbucket.org/maintainer/repo

需要自己用上面的命令创建upstream别名。这样可以简单地保持本地仓库和正式仓库的同步更新。 注意,如果上游仓库需要认证(比如不是开源的),你需要提供用户:

git remote add upstream https://user@bitbucket.org/maintainer/repo.git

在克隆和pull正式仓库时,需要提供用户的密码。

开发者开发自己的功能

在刚克隆的本地仓库中,开发者可以像其它工作流一样的编辑代码、提交修改和新建分支:

git checkout -b some-feature
# Edit some code
git commit -a -m "Add first draft of some feature"

所有的修改都是私有的直到push到自己公开仓库中。如果正式项目已经往前走了,可以用git pull命令获得:

git pull upstream master

由于开发都应该在专门的功能分支上开发,pull操作会是快进合并

开发者发布自己的功能

一旦开发者准备好了分享新功能,需要做二件事。 首先,通过push他的贡献代码到自己的公开仓库中,让其它的开发者都可以访问到。 他的origin远程别名应该已经有了,所以要做就是:

git push origin feature-branch

和之前的工作流的差异就在于,origin远程别名指向开发者自己的服务端仓库,而不是正式仓库。

然后,开发者要通知项目维护者,想合并他的新功能到正式库中。 BitbucketStash提供了Pull Request按钮,弹出表单让你指定哪个分支要合并到正式仓库。 一般你会想集合你的功能分支到上游远程仓库的master分支。

项目维护者集成开发者的功能

当项目维护者收到pull request,他要做的是决定是否集成到正式代码库中。有二种方式来做:

  1. 直接在pull request中查看代码
  2. pull代码到他自己的本地仓库,再手动合并

第一种做法更简单,维护者可以在GUI中查看变更的差异,做评注和执行合并。 但如果出现了合并冲突,需要第二种做法来解决。这种情况下,维护者需要从开发者服务端仓库上fetch功能分支, 合并到他本地的master分支,解决冲突:

git fetch https://bitbucket.org/user/repo feature-branch
# Inspect the changes
git checkout master
git merge FETCH_HEAD

变更集成到本地的master分支后,维护者要push变更到服务器上的正式仓库,这样其它的开发者都能访问到:

git push origin master

注意维护者的origin是指向他自己公有仓库的,即是项目的正式代码库。到此,开发者的贡献完全集成到了项目中。

开发者和正式仓库做同步

由于正式代码库往前走了,其它的开发需要做同步:

git pull upstream master

🍺 下一站

如果你之前是使用SVNForking工作流可能看起来像是一个激进的范式切换paradigm shift。 但不要害怕,这个工作流实际上就是在功能分支工作流之上引入另一个抽象层。 不是直接通过单个中央仓库来分享分支,而是把贡献代码发布到开发者自己的服务端仓库中。

示例中解释了,一个贡献如何从一个开发者流到正式的master分支中,但同样的方法可以把贡献集成到任一个仓库中。 比如,如果团队的几个人协作实现一个功能,可以在开发之间用相同的方法分享变更,不涉及路面仓库。

这使得Forking工作流对于松散组织的团队来说是个很强大的工具。任一开发者可以方便地和另一开发者分享变更,任何分支都能有效地合并到正式代码库。


« Gitflow工作流    Pull Requests »