5.6. Gerrit 代码审核服务器
谷歌Android开源项目在Git的使用上有两个重要的创新,一个是为多版本库协同而引入的repo,在前面第25章已经详细讨论过。另外一个重要的创新就是Gerrit——代码审核服务器。Gerrit为Git引入的代码审核是强制性的,就是说除非特别的授权设置,向Git版本库的推送(Push)必须要经过Gerrit服务器,修订必须经过代码审核的一套工作流之后,才可能经批准并纳入正式代码库中。
首先贡献者的代码通过Git命令(或repo封装)推送到Gerrit管理下的Git版本库,推送的提交转化为一个一个的代码审核任务,审核任务可以通过refs/changes/
下的引用访问到。代码审核者可以通过Web界面查看审核任务、代码变更,通过Web界面做出通过代码审核或者打回等决定。测试者也可以通过refs/changes/
之下的引用获取(fetch)修订对其进行测试,如果测试通过就可以将该评审任务设置为校验通过(verified)。最后经过了审核和校验的修订可以通过Gerrit界面中提交动作合并到版本库对应的分支中。
Android项目网站上有一个代码贡献流程图,详细的介绍了Gerrit代码审核服务器的工作流程。翻译后的工作流程图见图32-1。
图32-1:Gerrit代码审核工作流
5.6.1. Gerrit的实现原理
Gerrit更准确的说应该称为Gerrit2。因为Android项目最早使用的评审服务器Gerrit不是今天这个样子,最早版本的Gerrit是用Python开发运行于Google App Engine上,是从Python之父Guido van Rossum开发的Rietveld分支而来。在这里要讨论的Gerrit实为Gerrit2,是用Java语言实现的。从这里(http://code.google.com/p/gerrit/wiki/Background)可以看到Gerrit更为详尽的发展历史。
5.6.1.1. SSH协议的Git服务器
Gerrit本身基于SSH协议实现了一套Git服务器,这样就可以对Git数据推送进行更为精确的控制,为强制审核的实现建立了基础。
Gerrit提供的Git服务的端口并非标准的22端口,缺省是29418端口。可以访问Gerrit的Web界面得到这个端口。对Android项目的代码审核服务器,访问https://review.source.android.com/ssh_info就可以查看到Git服务的服务器域名和开放的端口。下面用curl命令查看网页的输出。
$ curl -L -k http://review.source.android.com/ssh_info review.source.android.com 29418
5.6.1.2. 特殊引用refs/for/<branch-name>
和refs/changes/nn/<review-id>/m
Gerrit的Git服务器,禁止用户向refs/heads
命名空间下的引用执行推送(除非特别的授权),即不允许用户直接向分支进行提交。为了让开发者能够向Git服务器提交修订,Gerrit的Git服务器只允许用户向特殊的引用refs/for/<branch-name>
下执行推送,其中<branch-name>
即为开发者的工作分支。向refs/for/<branch-name>
命名空间下推送并不会在其中创建引用,而是为新的提交分配一个ID,称为review-id ,并为该review-id的访问建立如下格式的引用refs/changes/nn/<review-id>/m
,其中:
- review-id为Gerrit为评审任务顺序分配的全局唯一的号码。
nn
为review-id的后两位数,位数不足用零补齐。即nn
为review-id除以100的余数。m
为修订号,该review-id的首次提交修订号为1,如果该修订被打回,重新提交修订号会自增。
5.6.1.3. Git库的钩子脚本hooks/commit-msg
为了保证已经提交审核的修订通过审核入库后,被别的分支拣选(cherry-pick)后再推送至服务器时不会产生新的重复的评审任务,Gerrit设计了一套方法,即要求每个提交包含唯一的Change-Id,这个Change-Id因为出现在日志中,当执行拣选时也会保持,Gerrit一旦发现新的提交包含了已经处理过的Change-Id
,就不再为该修订创建新的评审任务和review-id,而直接将提交入库。
为了实现Git提交中包含唯一的Change-Id,Gerrit提供了一个钩子脚本,放在开发者本地Git库中(hooks/commit-msg)。这个钩子脚本在用户提交时自动在提交说明中创建以“Change-Id:”及包含git hash-object命令产生的哈希值的唯一标识。当Gerrit获取到用户向refs/for/<branch-name>
推送的提交中包含“Change-Id: I...”的变更ID,如果该Change-Id之前没有见过,会创建一个新的评审任务并分配新的review-id,并在Gerrit的数据库中保存Change-Id和review-id的关联。
如果当用户的提交因为某种原因被要求打回重做,开发者修改之后重新推送到Gerrit时就要注意在提交说明中使用相同的“Change-Id”(使用--amend
提交即可保持提交说明),以免创建新的评审任务,还要在推送时将当前分支推送到refs/changes/nn/review-id/m
中。其中nn
和review-id
和之前提交的评审任务的修订相同,m
则要人工选择一个新的修订号。
以上说起来很复杂,但是在实际操作中只要使用repo这一工具,就相对容易多了。
5.6.1.4. 其余一切交给Web
Gerrit另外一个重要的组件就是Web服务器,通过Web服务器实现对整个评审工作流的控制。关于Gerrit工作流,参见在本章开头出现的Gerrit工作流程图。
感受一下Gerrit的魅力?直接访问Android项目的Gerrit网站:https://review.source.android.com/。会看到如图32-2的界面。
图32-2:Android项目代码审核网站
Android项目的评审网站,匿名即可访问。点击菜单中的“Merged”显示了已经通过评审合并到代码库中的审核任务。图32-3中显示的就是Andorid一个已经合并到代码库中的历史评审任务。
图32-3:Android项目的16993号评审
从图32-3可以看出:
- URL中显示的评审任务编号为16993。
- 该评审任务的Change-Id以字母
I
开头,包含了一个唯一的40位SHA1哈希。 - 整个评审任务有三个人参与,一个人进行了检查(verify),两个人进行了代码审核。
- 该评审任务的状态为已合并:“merged”。
- 该评审任务总共包含两个补丁集:Patch set 1和Patch set 2。
- 补丁集的下载方法是:repo download platform/sdk 16993/2。
如果使用repo
命令获取补丁集是非常方便的,因为封装后的repo
屏蔽掉了Gerrit的一些实现细节,例如补丁集在Git库中的存在位置。如前所述,补丁集实际保存在refs/changes
命名空间下。使用git ls-remote命令,从Gerrit维护的代码库中可以看到补丁集对应的引用名称。
$ git ls-remote \ ssh://review.source.android.com:29418/platform/sdk \ refs/changes/93/16993* 5fb1e79b01166f5192f11c5f509cf51f06ab023d refs/changes/93/16993/1 d342ef5b41f07c0202bc26e2bfff745b7c86d5a7 refs/changes/93/16993/2
接下来就来介绍一下Gerrit服务器的部署和使用方法。
5.6.2. 架设Gerrit的服务器
下载war包
Gerrit是由Java开发的,封装为一个war包:gerrit.war
,安装非常简洁。如果需要从源码编译出war包,可以参照文档:http://gerrit.googlecode.com/svn/documentation/2.1.5/dev-readme.html。不过最简单的就是从Google Code上直接下载编译好的war包。
从下面的地址下载Gerrit的war包:http://code.google.com/p/gerrit/downloads/list。在下载页面会有一个文件名类似Gerrit-x.x.x.war
的war包,这个文件就是Gerrit的全部。示例中使用的是2.1.5.1版本,把下载的Gerrit-2.1.5.1.war
包重命名为Gerrit.war。下面的介绍就是基于这个版本。
数据库选择
Gerrit需要数据库来维护账户信息、跟踪评审任务等。目前支持的数据库类型有PostgreSQL、MySQL以及嵌入式的H2数据库。
选择使用默认的H2内置数据库是最简单的,因为这样无须任何设置。如果想使用更为熟悉的PostgreSQL或者MySQL,则预先建立数据库。
对于PostgreSQL,在数据库中创建一个用户gerrit
,并创建一个数据库reviewdb。
createuser -A -D -P -E gerrit createdb -E UTF-8 -O gerrit reviewdb
对于MySQL,在数据库中创建一个用户gerrit并为其设置口令(不要真如下面的将口令置为secret
),并创建一个数据库reviewdb。
$ mysql -u root -p mysql> CREATE USER 'gerrit'@'localhost' IDENTIFIED BY 'secret'; mysql> CREATE DATABASE reviewdb; mysql> ALTER DATABASE reviewdb charset=latin1; mysql> GRANT ALL ON reviewdb.* TO 'gerrit'@'localhost'; mysql> FLUSH PRIVILEGES;
以一个专用用户帐号执行安装
在系统中创建一个专用的用户帐号如:gerrit。以该用户身份执行安装,将Gerrit的配置文件、内置数据库、war包等都自动安装在该用户主目录下的特定目录中。
$ sudo adduser gerrit $ sudo su gerrit $ cd ~gerrit $ java -jar gerrit.war init -d review_site
在安装过程中会提问一系列问题。
创建相关目录。
缺省Grerit在安装用户主目录下创建目录
review_site
并把相关文件安装在这个目录之下。Git版本库的根路径缺省位于此目录之下的git
目录中。*** Gerrit Code Review 2.1.5.1 *** Create '/home/gerrit/review_site' [Y/n]? *** Git Repositories *** Location of Git repositories [git]:
选择数据库类型。
选择H2数据库是简单的选择,无须额外的配置。
*** SQL Database *** Database server type [H2/?]:
设置Gerrit Web界面认证的类型。
缺省为openid,即使用任何支持OpenID的认证源(如Google、Yahoo)进行身份认证。此模式支持用户自建帐号,当用户通过OpenID认证源的认证后,Gerrit会自动从认证源获取相关属性如用户全名和邮件地址等信息创建帐号。Android项目的Gerrit服务器即采用此认证模式。
如果有可用的LDAP服务器,那么ldap或者ldap_bind也是非常好的认证方式,可以直接使用LDAP中的已有帐号进行认证,不过此认证方式下Gerrit的自建帐号功能关闭。此安装示例选择的就是LDAP认证方式。
http认证也是可选的认证方式,此认证方式需要配置Apache的反向代理并在Apache中配置Web站点的口令认证,通过口令认证后Gerrit在创建帐号的过程中会询问用户的邮件地址并发送确认邮件。
*** User Authentication *** Authentication method [OPENID/?]: ? Supported options are: openid http http_ldap ldap ldap_bind development_become_any_account Authentication method [OPENID/?]: ldap LDAP server [ldap://localhost]: LDAP username : Account BaseDN : dc=foo,dc=bar Group BaseDN [dc=foo,dc=bar]:
发送邮件设置。
缺省使用本机的SMTP发送邮件。
*** Email Delivery *** SMTP server hostname [localhost]: SMTP server port [(default)]: SMTP encryption [NONE/?]: SMTP username :
Java相关设置。
使用OpenJava和Sun Java均可。Gerrit的war包要复制到
review_site/bin
目录中。*** Container Process *** Run as [gerrit]: Java runtime [/usr/lib/jvm/java-6-sun-1.6.0.21/jre]: Copy gerrit.war to /home/gerrit/review_site/bin/gerrit.war [Y/n]? Copying gerrit.war to /home/gerrit/review_site/bin/gerrit.war
SSH服务相关设置。
Gerrit的基于SSH协议的Git服务非常重要,缺省的端口为29418。换做其他端口也无妨,因为repo可以自动探测到该端口。
*** SSH Daemon *** Listen on address [*]: Listen on port [29418]: Gerrit Code Review is not shipped with Bouncy Castle Crypto v144 If available, Gerrit can take advantage of features in the library, but will also function without it. Download and install it now [Y/n]? Downloading http://www.bouncycastle.org/download/bcprov-jdk16-144.jar ... OK Checksum bcprov-jdk16-144.jar OK Generating SSH host key ... rsa... dsa... done
HTTP服务相关设置。
缺省启用内置的HTTP服务器,端口为8080,如果该端口被占用(如Tomcat),则需要更换为其他端口,否则服务启动失败。如下例就换做了8888端口。
*** HTTP Daemon *** Behind reverse proxy [y/N]? y Proxy uses SSL (https://) [y/N]? y Subdirectory on proxy server [/]: /gerrit Listen on address [*]: Listen on port [8081]: Canonical URL [https://localhost/gerrit]: Initialized /home/gerrit/review_site
启动Gerrit服务
Gerrit服务正确安装后,运行Gerrit启动脚本启动Gerrit服务。
$ /home/gerrit/review_site/bin/gerrit.sh start Starting Gerrit Code Review: OK
服务正确启动之后,会看到Gerrit服务打开两个端口,这两个端口是在Gerrit安装时指定的。您的输出和下面的示例可能略有不同。
$ sudo netstat -ltnp | grep -i gerrit tcp 0 0 0.0.0.0:8081 0.0.0.0:* LISTEN 26383/GerritCodeRev tcp 0 0 0.0.0.0:29418 0.0.0.0:* LISTEN 26383/GerritCodeRev
设置Gerrit服务开机自动启动
Gerrit服务的启动脚本支持start、stop、restart参数,可以作为init脚本开机自动执行。
$ sudo ln -snf \ /home/gerrit/review_site/bin/gerrit.sh \ /etc/init.d/gerrit.sh $ sudo ln -snf ../init.d/gerrit.sh /etc/rc2.d/S90gerrit $ sudo ln -snf ../init.d/gerrit.sh /etc/rc3.d/S90gerrit
服务自动启动脚本/etc/init.d/gerrit.sh
需要通过/etc/default/gerritcodereview
提供一些缺省配置。以下面内容创建该文件。
GERRIT_SITE=/home/gerrit/review_site NO_START=0
Gerrit认证方式的选择
如果是开放服务的Gerrit服务,使用OpenId认证是最好的方法,就像谷歌Android项目的代码审核服务器配置的那样。任何人只要在具有OpenId provider的网站上(如Google、Yahoo等)具有帐号,就可以直接通过OpenId注册,Gerrit会在用户登录OpenId provider网站成功后,自动获取(经过用户的确认)用户在OpenId provider站点上的部分注册信息(如用户全名或者邮件地址)在Gerrit上自动为用户创建帐号。
如果架设有LDAP服务器,并且用户帐号都在LDAP中进行管理,那么采用LDAP认证也是非常好的方法。登录时提供的用户名和口令通过LDAP服务器验证之后,Gerrit会自动从LDAP服务器中获取相应的字段属性,为用户创建帐号。创建的帐号的用户全名和邮件地址因为来自于LDAP,因此不能在Gerrit更改,但是用户可以注册新的邮件地址。我在配置LDAP认证时遇到了一个问题就是创建帐号的用户全名是空白,这是因为在LDAP相关的字段没有填写的原因。如果LDAP服务器使用的是OpenLDAP,Gerrit会从displayName字段获取用户全名,如果使用Active Directory则用givenName和sn字段的值拼接形成用户全名。
Gerrit还支持使用HTTP认证,这种认证方式需要架设Apache反向代理,在Apache中配置HTTP认证。当用户访问Gerrit网站首先需要通过Apache配置的HTTP Basic Auth认证,当Gerrit发现用户已经登录后,会要求用户确认邮件地址。当用户邮件地址确认后,再填写其他必须的字段完成帐号注册。HTTP认证方式的缺点除了在口令文件管理上需要管理员手工维护比较麻烦之外,还有一个缺点就是用户一旦登录成功后,想退出登录或者更换其他用户帐号登录变得非常麻烦,除非关闭浏览器。关于切换用户有一个小窍门:例如Gerrit登录URL为https://server/gerrit/login/,则用浏览器访问https://nobody:wrongpass@server/gerrit/login/,即用错误的用户名和口令覆盖掉浏览器缓存的认证用户名和口令,这样就可以重新认证了。
在后面的Gerrit演示和介绍中,为了设置帐号的方便,使用了HTTP认证,因此下面再介绍一下HTTP认证的配置方法。
配置Apache代理访问Gerrit
缺省Gerrit的Web服务端口为8080或者8081,通过Apache的反向代理就可以使用标准的80(http)或者443(https)来访问Gerrit的Web界面。
ProxyRequests Off ProxyVia Off ProxyPreserveHost On <Proxy *> Order deny,allow Allow from all </Proxy> ProxyPass /gerrit/ http://127.0.0.1:8081/gerrit/
如果要配置Gerrit的http认证,则还需要在上面的配置中插入Http Basic认证的设置。
<Location /gerrit/login/> AuthType Basic AuthName "Gerrit Code Review" Require valid-user AuthUserFile /home/gerrit/review_site/etc/gerrit.passwd </Location>
在上面的配置中,指定了口令文件的位置:/home/gerrit/review_site/etc/gerrit.passwd
。可以用htpasswd命令维护该口令文件。
$ touch /home/gerrit/review_site/etc/gerrit.passwd $ htpasswd -m /home/gerrit/review_site/etc/gerrit.passwd jiangxin New password: Re-type new password: Adding password for user jiangxin
至此为止,Gerrit服务安装完成。在正式使用Gerrit之前,先来研究一下Gerrit的配置文件,以免安装过程中遗漏或错误的设置影响使用。
5.6.3. Gerrit的配置文件
Gerrit的配置文件保存在部署目录下的etc/gerrit.conf
文件中。如果对安装时的配置不满意,可以手工修改配置文件,重启Gerrit服务即可。
全部采用缺省配置时的配置文件:
[gerrit] basePath = git canonicalWebUrl = http://localhost:8080/ [database] type = H2 database = db/ReviewDB [auth] type = OPENID [sendemail] smtpServer = localhost [container] user = gerrit javaHome = /usr/lib/jvm/java-6-openjdk/jre [sshd] listenAddress = *:29418 [httpd] listenUrl = http://*:8080/ [cache] directory = cache
如果采用LDAP认证,下面的配置文件片断配置了一个支持匿名绑定的LDAP服务器配置。
[auth] type = LDAP [ldap] server = ldap://localhost accountBase = dc=foo,dc=bar groupBase = dc=foo,dc=bar
如果采用MySQL而非缺省的H2数据库,下面的配置文件显示了相关配置。
[database] type = MYSQL hostname = localhost database = reviewdb username = gerrit
LDAP绑定或者数据库连接的用户口令保存在:file:` etc/secure.config`文件中。
[database] password = secret
下面的配置将Web服务架设在Apache反向代理的后面。
[httpd] listenUrl = proxy-https://*:8081/gerrit
5.6.4. Gerrit的数据库访问
之所以要对数据库访问多说几句,是因为一些对Gerrit的设置往往在Web界面无法配置,需要直接修改数据库,而大部分用户在安装Gerrit时都会选用内置的H2数据库,如何操作H2数据库可能大部分用户并不了解。
实际上无论选择何种数据库,Gerrit都提供了两种数据库操作的命令行接口。第一种方法是在服务器端调用gerrit.war
包中的命令入口,另外一种方法是远程SSH调用接口。
对于第一种方法,需要在服务器端执行,而且如果使用的是H2内置数据库还需要先将Gerrit服务停止。先以安装用户身份进入Gerrit部署目录下,在执行命令调用gerrit.war
包,如下:
$ java -jar bin/gerrit.war gsql Welcome to Gerrit Code Review 2.1.5.1 (H2 1.2.134 (2010-04-23)) Type '\h' for help. Type '\r' to clear the buffer. gerrit>
当出现“gerrit>
”提示符时,就可以输入SQL语句操作数据库了。
第一种方式需要登录到服务器上,而且操作H2数据库时还要预先停止服务,显然很不方便。但是这种方法也有存在的必要,就是不需要认证,尤其是在管理员帐号尚未建立之前就可以查看和更改数据库。
当在Gerrit上注册了第一个帐号,即拥有了管理员帐号,正确为该帐号配置公钥之后,就可以访问Gerrit提供的SSH登录服务。Gerrit的SSH协议提供第二个访问数据库的接口。下面的命令就是用管理员公钥登录Gerrit的SSH服务器,操作数据库。虽然演示用的是本机地址(localhost),但是操作远程服务器也是可以的,只要拥有管理员授权。
$ ssh -p 29418 localhost gerrit gsql Welcome to Gerrit Code Review 2.1.5.1 (H2 1.2.134 (2010-04-23)) Type '\h' for help. Type '\r' to clear the buffer. gerrit>
即连接Gerrit的SSH服务,运行命令gerrit gsql。当连接上数据库管理接口后,便出现“gerrit>
”提示符,在该提示符下可以输入SQL命令。下面的示例中使用的数据库后端为H2内置数据库。
可以输入show tables
命令显示数据库列表。
gerrit> show tables; TABLE_NAME | TABLE_SCHEMA ----------------------------+------------- ACCOUNTS | PUBLIC ACCOUNT_AGREEMENTS | PUBLIC ACCOUNT_DIFF_PREFERENCES | PUBLIC ACCOUNT_EXTERNAL_IDS | PUBLIC ACCOUNT_GROUPS | PUBLIC ACCOUNT_GROUP_AGREEMENTS | PUBLIC ACCOUNT_GROUP_MEMBERS | PUBLIC ACCOUNT_GROUP_MEMBERS_AUDIT | PUBLIC ACCOUNT_GROUP_NAMES | PUBLIC ACCOUNT_PATCH_REVIEWS | PUBLIC ACCOUNT_PROJECT_WATCHES | PUBLIC ACCOUNT_SSH_KEYS | PUBLIC APPROVAL_CATEGORIES | PUBLIC APPROVAL_CATEGORY_VALUES | PUBLIC CHANGES | PUBLIC CHANGE_MESSAGES | PUBLIC CONTRIBUTOR_AGREEMENTS | PUBLIC PATCH_COMMENTS | PUBLIC PATCH_SETS | PUBLIC PATCH_SET_ANCESTORS | PUBLIC PATCH_SET_APPROVALS | PUBLIC PROJECTS | PUBLIC REF_RIGHTS | PUBLIC SCHEMA_VERSION | PUBLIC STARRED_CHANGES | PUBLIC SYSTEM_CONFIG | PUBLIC TRACKING_IDS | PUBLIC (27 rows; 65 ms)
输入show columns
命令显示数据库的表结构。
gerrit> show columns from system_config; FIELD | TYPE | NULL | KEY | DEFAULT ---------------------------+--------------+------+-----+-------- REGISTER_EMAIL_PRIVATE_KEY | VARCHAR(36) | NO | | '' SITE_PATH | VARCHAR(255) | YES | | NULL ADMIN_GROUP_ID | INTEGER(10) | NO | | 0 ANONYMOUS_GROUP_ID | INTEGER(10) | NO | | 0 REGISTERED_GROUP_ID | INTEGER(10) | NO | | 0 WILD_PROJECT_NAME | VARCHAR(255) | NO | | '' BATCH_USERS_GROUP_ID | INTEGER(10) | NO | | 0 SINGLETON | VARCHAR(1) | NO | PRI | '' (8 rows; 52 ms)
关于H2数据库更多的SQL语法,参考:http://www.h2database.com/html/grammar.html。
下面开始介绍Gerrit的使用。
5.6.5. 立即注册为Gerrit管理员
第一个Gerrit账户自动成为权限最高的管理员,因此Gerrit安装完毕后的第一件事情就是立即注册或者登录,以便初始化管理员帐号。下面的示例是在本机(localhost)以HTTP认证方式架设的Gerrit审核服务器。当第一次访问的时候,会弹出非常眼熟的HTTP Basic Auth认证界面,如图32-4。
图32-4:Http Basic Auth 认证界面
输入正确的用户名和口令登录后,系统自动创建ID为1000000的帐号,该帐号是第一个注册的帐号,会自动该被赋予管理员身份。因为使用的是HTTP认证,用户的邮件地址等个人信息尚未确定,因此登录后首先进入到个人信息设置界面。如图32-5。
图32-5:Gerrit第一次登录后的个人信息设置界面
在图32-5中可以看到在菜单中有“Admin”菜单项,说明当前登录的用户被赋予了管理员权限。在图32-5的联系方式确认对话框中有一个注册新邮件地址的按钮,点击该按钮弹出邮件地址录入对话框,如图32-6。
图32-6:输入个人的邮件地址
必须输入一个有效的邮件地址以便能够收到确认邮件。这个邮件地址非常重要,因为Git代码提交时在提交说明中出现的邮件地址需要和这个地址一致。当填写了邮件地址后,会收到一封确认邮件,点击邮件中的确认链接会重新进入到Gerrit帐号设置界面,如图32-7。
图32-7:邮件地址确认后进入Gerrit界面
在Full Name
字段输入用户名,点击保存更改后,右上角显示的“Anonymous Coward” 就会显示为登录用户的姓名和邮件地址。
接下来需要做的最重要的一件事就是配置公钥(如图32-8)。通过该公钥,注册用户可以通过SSH协议向Gerrit的Git服务器提交,如果具有管理员权限还能够远程管理Gerrit服务器。
图32-8:Gerrit的SSH公钥设置界面
在文本框中粘贴公钥。关于如何生成和管理公钥,参见第29章“使用SSH协议”相关内容。
点击“Add”按钮,完成公钥的添加。添加的公钥就会显示在列表中(如图32-9)。一个用户可以添加多个公钥。
图32-9:用户的公钥列表
点击左侧的“Groups”(用户组)菜单项,可以看到当前用户所属的分组,如图32-10。
图32-10:Gerrit用户所属的用户组
第一个注册的用户同时属于三个用户组,一个是管理员用户组(Administrators),另外两个分别是Anonymous Users(任何用户)和Registered Users(注册用户)。
5.6.6. 管理员访问SSH的管理接口
当在Gerrit个人配置界面中设置了公钥之后,就可以连接Gerrit的SSH服务器执行命令,示例使用的是本机(localhost),其实远程IP地址一样可以。只是对于远程主机需要确认端口不要被防火墙拦截,Gerrit的SSH服务器使用特殊的端口,缺省是29418。
任何用户都可以通过SSH连接执行gerrit ls-projects命令查看项目列表。下面的命令没有输出,是因为项目尚未建立。
$ ssh -p 29418 localhost gerrit ls-projects
可以执行scp命令从Gerrit的SSH服务器中拷贝文件。
$ scp -P 29418 -p -r localhost:/ gerrit-files $ find gerrit-files -type f gerrit-files/bin/gerrit-cherry-pick gerrit-files/hooks/commit-msg
可以看出Gerrit服务器提供了两个文件可以通过scp下载,其中commit-msg
脚本文件应该放在用户本地Git库的钩子目录中以便在生成的提交中包含唯一的Change-Id。在之前的Gerrit原理中介绍过。
除了普通用户可以执行的命令外,管理员还可以通过SSH连接执行Gerrit相关的管理命令。例如之前介绍的管理数据库:
$ ssh -p 29418 localhost gerrit gsql Welcome to Gerrit Code Review 2.1.5.1 (H2 1.2.134 (2010-04-23)) Type '\h' for help. Type '\r' to clear the buffer. gerrit>
此外管理员还可以通过SSH连接执行帐号创建,项目创建等管理操作,可以执行下面的命令查看帮助信息。
$ ssh -p 29418 localhost gerrit --help gerrit COMMAND [ARG ...] [--] [--help (-h)] -- : end of options --help (-h) : display this help text Available commands of gerrit are: approve create-account create-group create-project flush-caches gsql ls-projects query receive-pack replicate review set-project-parent show-caches show-connections show-queue stream-events See 'gerrit COMMAND --help' for more information.
更多的帮助信息,还可以参考Gerrit版本库中的帮助文件:Documentation/cmd-index.html
。
5.6.7. 创建新项目
一个Gerrit项目对应于一个同名的Git库,同时拥有一套可定制的评审流程。创建一个新的Gerrit项目就会在对应的版本库根目录下创建Git库。管理员可以使用命令行创建新项目。
$ ssh -p 29418 localhost gerrit create-project --name new/project
当执行gerrit ls-projects命令,可以看到新项目创建已经成功创建。
$ ssh -p 29418 localhost gerrit ls-projects new/project
在Gerrit的Web管理界面,也可以看到新项目已经建立,如图32-11。
图32-11:Gerrit中项目列表
在项目列表中可以看到除了新建的new/project
项目之外还有一个名为“– All Projects –”的项目,其实它并非一个真实存在的项目,只是为了项目授权管理的方便。即在“– All Projects –” 中建立的项目授权能够被其他项目共享。
在服务器端也可以看到Gerrit部署中版本库根目录下已经有同名的Git版本库被创建。
$ ls -d /home/gerrit/review_site/git/new/project.git /home/gerrit/review_site/git/new/project.git
这个新的版本库刚刚初始化,尚未包括任何数据。是否可以通过git push向该版本库推送一些初始数据呢?下面用Gerrit的SSH协议克隆该版本库,并尝试向其推送数据。
$ git clone ssh://localhost:29418/new/project.git myproject Cloning into myproject... warning: You appear to have cloned an empty repository. $ cd myproject/ $ echo hello > readme.txt $ git add readme.txt $ git commit -m "initialized." [master (root-commit) 15a549b] initialized. 1 files changed, 1 insertions(+), 0 deletions(-) create mode 100644 readme.txt 09:58:54 jiangxin@hp:~/tmp/myproject$ git push origin master Counting objects: 3, done. Writing objects: 100% (3/3), 222 bytes, done. Total 3 (delta 0), reused 0 (delta 0) To ssh://localhost:29418/new/project.git ! [remote rejected] master -> master (prohibited by Gerrit) error: failed to push some refs to 'ssh://localhost:29418/new/project.git'
向Gerrit的Git版本库推送失败,远程Git服务器返回错误信息:“prohibited by Gerrit”。这是因为Gerrit缺省不允许直接向分支推送,而是需要向refs/for/<branch-name>
的特殊引用进行推送以便将提交转换为评审任务。
但是如果希望将版本库的历史提交不经审核直接推送到Gerrit维护的Git版本库中可以么?是的,只要通过Gerrit的管理界面为该项目授权:允许某个用户组(如Administrators组)的用户可以向分支推送。(注意该授权在推送完毕后尽快撤销,以免被滥用)
Gerrit的界面对用户非常友好(如图32-12)。例如在添加授权的界面中,只要在用户组的输入框中输入前几个字母,就会弹出用户组列表供选择。
图32-12:添加授权的界面
添加授权完毕后,项目“new/project”的授权列表就会出现新增的为Administrators管理员添加的“+2: Create Branch”授权,如图32-13。
图32-13:添加授权后的授权列表
因为已经为管理员分配了直接向refs/heads/*
引用推送的授权,这样就能够向Git版本库推送数据了。再执行一次推送任务,看看能否成功。
$ git push origin master Counting objects: 3, done. Writing objects: 100% (3/3), 222 bytes, done. Total 3 (delta 0), reused 0 (delta 0) To ssh://localhost:29418/new/project.git ! [remote rejected] master -> master (you are not committer jiangxin@ossxp.com) error: failed to push some refs to 'ssh://localhost:29418/new/project.git'
推送又失败了,但是服务器端返回的错误信息不同。上一次出错返回的是“prohibited by Gerrit”,而这一次返回的错误信息是“you are not committer”。
这是为什么呢?看看提交日志:
$ git log --pretty=full commit 15a549bac6bd03ad36e643984fed554406480b2c Author: Jiang Xin <jiangxin@ossxp.com> Commit: Jiang Xin <jiangxin@ossxp.com> initialized.
提交者(Commit)为“Jiang Xin <jiangxin@ossxp.com>”,而Gerrit中注册的用户的邮件地址是“jiangxin@moon.ossxp.com”,两者之间的不一致,导致Gerrit再一次拒绝了提交。如果再到Gerrit看一下new/project
的权限设置,会看到这样一条授权:
Category Group Name Reference Name Permitted Range ======== ========== ============== =============== Forge Identity Registered Users refs/* +1: Forge Author Identity
这条授权的含义是提交中的Author字段不进行邮件地址是否注册的检查,但是要对Commit字段进行邮件地址检查。如果增加一个更高级别的“Forge Identity”授权,也可以忽略对Committer的邮件地址检查,但是尽量不要对授权进行非必须的改动,因为在提交的时候使用注册的邮件地址是一个非常好的实践。
下面就通过git config命令修改提交时所用的邮件地址,和Gerrit注册时用的地址保持一致。然后用--amend
参数重新执行提交以便让修改后的提交者邮件地址在提交中生效。
$ git config user.email jiangxin@moon.ossxp.com $ git commit --amend -m initialized [master 82c8fc3] initialized Author: Jiang Xin <jiangxin@ossxp.com> 1 files changed, 1 insertions(+), 0 deletions(-) create mode 100644 readme.txt $ git push origin master Counting objects: 3, done. Writing objects: 100% (3/3), 233 bytes, done. Total 3 (delta 0), reused 0 (delta 0) To ssh://localhost:29418/new/project.git * [new branch] master -> master
看这次提交成功了!之所以成功,是因为提交者的邮件地址更改了。看看重新提交的日志,可以发现作者(Author)和提交者(Commit)的邮件地址的不同,Commit
字段的邮件地址和注册时使用的邮件地址相同。
$ git log --pretty=full commit 82c8fc3805d57cc0d17d58e1452e21428910fd2d Author: Jiang Xin <jiangxin@ossxp.com> Commit: Jiang Xin <jiangxin@moon.ossxp.com> initialized
注意,版本库初始化完成之后,应尽快把为项目新增的“Push Branch”类型的授权删除,对新的提交强制使用Gerrit的评审流程。
5.6.8. 从已有Git库创建项目
如果已经拥有很多版本库,希望从这些版本库创建Gerrit项目,如果像上面介绍的那样一个一个的创建项目,再执行git push命令推送已经包含历史数据的版本库,将是十分麻烦的事情。那么有没有什么简单的办法呢?可以通过下面的步骤,实现多项目的快速创建。
首先将已有版本库创建到Gerrit的版本库根目录下。注意版本库名称将会成为项目名(除去.git
后缀),而且创建(或克隆)的版本库应为裸版本库,即使用--bare
参数创建。
例如在Gerrit的Git版本库根目录下创建名为hello.git
的版本库。下面的示例中我偷了一下懒,直接从new/project
克隆到hello.git
。:)
$ git clone --mirror \ /home/gerrit/review_site/git/new/project.git \ /home/gerrit/review_site/git/hello.git
这时查看版本库列表,却看不到新建立的名为hello.git
的Git库出现在项目列表中。
$ ssh -p 29418 localhost gerrit ls-projects new/project
可以通过修改Gerrit数据库来注册新项目,即连接到Gerrit数据库,输入SQL插入语句。
$ ssh -p 29418 localhost gerrit gsql Welcome to Gerrit Code Review 2.1.5.1 (H2 1.2.134 (2010-04-23)) Type '\h' for help. Type '\r' to clear the buffer. gerrit> INSERT INTO projects -> (use_contributor_agreements ,submit_type ,name) -> VALUES -> ('N' ,'M' ,'hello'); UPDATE 1; 1 ms gerrit>
注意SQL语句中的项目名称是版本库名称除去.git
后缀的部分。在数据库插入数据后,再来查看项目列表就可以看到新注册的项目了。
$ ssh -p 29418 localhost gerrit ls-projects hello new/project
可以登录到Gerrit项目对新建立的项目进行相关设置。例如修改项目的说明,项目的提交策略,是否要求提交说明中必须包含“Signed-off-by”信息等,如图32-14。
图32-14:项目基本设置
这种通过修改数据库从已有版本库创建项目的方法适合大批量的项目创建。下面就对新建立的hello项目进行一次完整的Gerrit评审流程。
5.6.9. 定义评审工作流
刚刚安装好的Gerrit的评审工作流并不完整,还不能正常的开展评审工作,需要对项目授权进行设置以定制适合的评审工作流。
缺省安装的Gerrit中只内置了四个用户组,如表32-1所示。
表32-1:Gerrit内置用户组
用户组 说明 Administrators Gerrit 管理员 Anonymous Users 任何用户,登录或未登录 Non-Interactive Users Gerrit 中执行批处理的用户 Registered Users 任何登录用户
未登录的用户只属于Anonymous Users,登录用户则同时拥有Anonymous Users和Registered Users的权限。对于管理员则还拥有Administrators用户组权限。
查看全局(伪项目“– All Projects –”)的初始权限设置。会看到如表32-2一样的授权表格。
表32-2:Gerrit授权表格
编号 类别 用户组名称 引用名称 权限范围 1 Code Review Registered Users refs/heads/* -1: I would prefer that you didn’t submit this +1: Looks good to me, but someone else must approve 2 Forge Identity Registered Users refs/* +1: Forge Author Identity 3 Read Access Administrators refs/* +1: Read access 4 Read Access Anonymous Users refs/* +1: Read access 5 Read Access Registered Users refs/* +2: Upload permission
对此表格中的授权解读如下:
对于匿名用户:根据第4条授权策略,匿名用户能够读取任意版本库。
对于注册用户:根据第5条授权策略,注册用户具有比第四条授权高一个等级的权限,即注册用户除了具有读取版本库权限外,还可以向版本库的
refs/for/<branch-name>
引用推送,产生评审任务的权限。之所以这种可写的权限也放在“Read Access”类别中,是因为Git的写操作必须建立在拥有读权限之上,因此Gerrit将其与读取都放在“Read Access”归类之下,只不过更高一个级别。
对于注册用户:根据第2条授权策略,在向服务器推送提交的时候,忽略对提交中Author字段的邮件地址检查。这个在之前已经讨论过。
对于注册用户:根据第1条授权策略,注册用户具有代码审核的一般权限,即能够将评审任务设置为“+1”级别(看起来不错,但需要通过他人认可),或者将评审任务标记为“-1”,即评审任务没有通过不能提交。
对于管理员:根据第3条策略,管理员能够读取任意版本库。
上面的授权策略仅仅对评审流程进行了部分设置。如:提交能够进入评审流程,因为登录用户(注册用户)可以将提交以评审任务方式上传;注册用户可以将评审任务标记为“+1: 看起来不错,但需其他人认可”。但是没有人有权限可以将评审任务提交——合并到正式版本库中,即没人能够对评审任务做最终的确认及提交,因此评审流程是不完整的。
要想实现对评审最终确认的授权,有两种方法可以实现,一种是赋予特定用户Verified类别中的“+1: Verified”的授权,另外一个方法是赋予特定用户Code Review类别中更高级别的授权:“+2: Looks good to me, approved”。要想实现对经过确认的评审任务提交,还需要赋予特定用户Submit类别中的“+1: Submit”授权。
下面的示例中,创建两个新的用户组Reviewer和Verifier,并为其赋予相应的授权。
创建用户组,可以通过Web界面或者命令行。如果通过Web界面添加用户组,选择“Admin”菜单下的“Groups”子菜单,如图32-15。
图32-15:Gerrit用户组创建
输入用户组名称后,点击“Create Group”按钮。进入创建用户组后的设置页,如图32-16。
图32-16:Gerrit用户组设置页
注意到在用户设置页面中有一个Owners字段名称和用户组名称相同,实际上这是Gerrit关于用户组的一个特别的功能。一个用户组可以设置另外一个用户组为本用户组的Owners,属于Owners用户组的用户实际上相当于本用户组的管理者,可以添加用户、修改用户组名称等。不过一般最常用的设置是使用同名的用户组作为Owners。
在用户组设置页面的最下面,是用户组用户分配对话框,可以将用户分配到用户组中。注意Gerrit的用户组不能包含,即只能将用户分配到用户组中。
图32-17是添加了两个新用户组后的用户组列表:
图32-17:Gerrit用户组列表
接下来要为新的用户组授权,需要访问“Admin”菜单下的“Projects”子菜单,点击对应的项目进入权限编辑界面。为了简便起见,选择“– All Projects –”,对其授权的更改可以被所有其他的项目共享。图32-18是为Reviewer用户组建立授权过程的页面。
图32-18:为Reviewer用户组建立授权
分别为两个新建立的用户组分配授权,如表32-3所示。编号从6开始,是因为这里补充的授权是建立在前面的缺省授权列表的基础上的。
表32-3:新用户组权限分配表
编号 类别 用户组名称 引用名称 权限范围 6 Code Review Reviewer refs/* -2: Do not submit +2: Looks good to me, approved 7 Verified Verifier refs/* -1: Fails +1: Verified 8 Submit Verifier refs/* +1: Submit
这样,就为Gerrit所有的项目设定了可用的评审工作流。
5.6.10. Gerrit评审工作流实战
分别再注册两个用户帐号dev1@moon.ossxp.com
和dev2@moon.ossxp.com
,两个用户分别属于Reviewer用户组和Verifier用户组。这样Gerrit部署中就拥有了三个用户帐号,用帐号jiangxin
进行代码提交,用dev1
帐号对任务进行代码审核,用dev2
用户对审核任务进行最终的确认。
5.6.10.1. 开发者在本地版本库中工作
Repo是Gerrit的最佳伴侣,凡是需要和Gerrit版本库交互的工作都封装在repo命令中。关于repo的用法在上一部分的repo多版本库协同的章节中已经详细介绍了。这里只介绍开发者如何只使用Git命令来和Gerrit服务器交互。这样也可以更深入的理解repo和gerrit整合的机制。
首先克隆Gerrit管理的版本库,使用Gerrit提供的运行于29418端口的SSH协议。
$ git clone ssh://localhost:29418/hello.git Cloning into hello... remote: Counting objects: 3, done remote: Compressing objects: 100% (3/3) Receiving objects: 100% (3/3), done.
然后拷贝Gerrit服务器提供的commit-msg
钩子脚本。
$ cd hello $ scp -P 29418 -p localhost:/hooks/commit-msg .git/hooks/
别忘了修改Git配置中提交者的邮件地址,以便和Gerrit中注册的地址保持一致。不使用--global
参数调用git config可以只对本版本库的提交设定提交者邮件。
$ git config user.email jiangxin@moon.ossxp.com
然后修改readme.txt
文件,并提交。注意提交的时候使用了-s
参数,目的是在提交说明中加入“Signed-off-by:”标记,这在Gerrit提交中可能是必须的。
$ echo "gerrit review test" >> readme.txt $ git commit -a -s -m "readme.txt hacked." [master c65ab49] readme.txt hacked. 1 files changed, 1 insertions(+), 0 deletions(-)
查看一下提交日志,会看到其中有特殊的标签。
$ git log --pretty=full -1 commit c65ab490f6d3dc36429b8f1363b6191357202f2e Author: Jiang Xin <jiangxin@moon.ossxp.com> Date: Mon Nov 15 17:50:08 2010 +0800 readme.txt hacked. Change-Id: Id7c9d88ebf5dac2d19a7e0896289de1ae6fb6a90 Signed-off-by: Jiang Xin <jiangxin@moon.ossxp.com>
提交说明中出现了“Change-Id:”标签,这个标签是由钩子脚本commit-msg
自动生成的。至于这个标签的含义,在前面Gerrit的实现原理中介绍过。
好了,准备把这个提交PUSH到服务器上吧。
5.6.10.2. 开发者向审核服务器提交
由Gerrit控制的Git版本库不能直接提交,因为正确设置的Gerrit服务器,会拒绝用户直接向refs/heads/*
推送。
$ git status # On branch master # Your branch is ahead of 'origin/master' by 1 commit. # nothing to commit (working directory clean) $ git push Counting objects: 5, done. Writing objects: 100% (3/3), 332 bytes, done. Total 3 (delta 0), reused 0 (delta 0) To ssh://localhost:29418/hello.git ! [remote rejected] master -> master (prohibited by Gerrit) error: failed to push some refs to 'ssh://localhost:29418/hello.git'
直接推送就会出现遇到“prohibited by Gerrit”的错误。
正确的做法是向特殊的引用推送,这样Gerrit会自动将新提交转换为评审任务。
$ git push origin HEAD:refs/for/master Counting objects: 5, done. Writing objects: 100% (3/3), 332 bytes, done. Total 3 (delta 0), reused 0 (delta 0) To ssh://localhost:29418/hello.git * [new branch] HEAD -> refs/for/master
看到了么,向refs/for/master
推送成功。
5.6.10.3. 审核评审任务
以dev1
用户登录Gerrit网站,点击“All”菜单下的“Open”标签,可以新提交到Gerrit状态为Open的评审任务,如图32-19。
图32-19:Gerrit评审任务列表
点击该评审任务,显示关于此评审任务的详细信息,如图32-20。
图32-20:Gerrit评审任务概述
从URL地址栏可以看到该评审任务的评审编号为1。目前该评审任务有一个补丁集(Patch Set 1),可以点击“Diff All Side-by-Side”查看变更集,以决定该提交是否应该被接受。作为测试,先让此次提交通过代码审核,于是以dev1
用户身份点击“Review”按钮。
点击“Review”按钮后,弹出代码评审对话框,如图32-21。
图32-21:Gerrit任务评审对话框
选择“+2: Looks good to me, approved.”,点击按钮“Publish Comments”以通过评审。注意因为没有给dev1
用户(Reviewer用户组)授予Submit权限,因此此时dev1
还不能将此审核任务提交。
当dev1
用户做出通过评审的决定后,代码提交者jiangxin
会收到一封邮件,如图32-22。
图32-22:Gerrit通知邮件
5.6.10.4. 评审任务没有通过测试
下面以dev2
帐号登录Gerrit,查看处于打开状态的评审任务,如图32-23。会看到评审任务1的代码评审已经通过,但是尚未进行测试检查(Verify)。于是dev2
可以下载该补丁集,在本机进行测试。
图32-23:Gerrit评审任务显示
假设测试没有通过,dev2
用户点击该评审任务的“Review”按钮,重置该任务的评审状态,如图32-24。
图32-24:Gerrit评审任务未通过
注意到图32-24中dev2
用户的评审对话框有三个按钮,多出的“Publish and Submit”按钮是因为dev2
拥有Submit授权。dev2
用户在上面的对话框中,选择了“-1: Fails”,当点击“Publish Comments”按钮,该评审任务的评审记录被重置,同时提交者和其他评审参与者会收到通知邮件,如图32-25。
图32-25:Gerrit通知邮件:评审未通过
5.6.10.5. 重新提交新的补丁集
提交者收到代码被打回的邮件,一定很难过。不过这恰恰说明了这个软件过程已经相当的完善,现在发现问题总比在集成测试时甚至被客户发现要好的多吧。
根据评审者和检验者的提示,开发者对代码进行重新修改。下面的bugfix过程仅仅是一个简单的示例,bugfix没有这么简单的,对么?;-)
$ echo "fixed" >> readme.txt
重新修改后,需要使用--amend
参数进行提交,即使用前次提交的日志重新提交,这一点非常重要。因为这样就会对原提交说明中的“Change-Id:”标签予以原样保留,当再将新提交推送到服务器时,Gerrit不会为新提交生成新的评审任务编号而是会重用原有的任务编号,将新提交转化为老的评审任务的新的补丁集。
在执行git commit --amend时,可以修改提交说明,但是注意不要删除Change-Id标签,更不能修改它。
$ git add -u $ git commit --amend readme.txt hacked with bugfix. Change-Id: Id7c9d88ebf5dac2d19a7e0896289de1ae6fb6a90 Signed-off-by: Jiang Xin <jiangxin@moon.ossxp.com> # Please enter the commit message for your changes. Lines starting # with '#' will be ignored, and an empty message aborts the commit. # On branch master # Your branch is ahead of 'origin/master' by 1 commit. # # Changes to be committed: # (use "git reset HEAD^1 <file>..." to unstage) # # modified: readme.txt #
提交成功后,执行git ls-remote命令会看到Gerrit维护的Git库中只有一个评审任务(编号1),且该评审任务只有一个补丁集(Patch Set 1)。
$ git ls-remote origin 82c8fc3805d57cc0d17d58e1452e21428910fd2d HEAD c65ab490f6d3dc36429b8f1363b6191357202f2e refs/changes/01/1/1 82c8fc3805d57cc0d17d58e1452e21428910fd2d refs/heads/master
把修改后的提交推送到Gerrit管理下的Git版本库中。注意依旧推送到refs/for/master
引用中。
$ git push origin HEAD:refs/for/master Counting objects: 5, done. Writing objects: 100% (3/3), 353 bytes, done. Total 3 (delta 0), reused 0 (delta 0) To ssh://localhost:29418/hello.git * [new branch] HEAD -> refs/for/master
推送成功后,再执行git ls-remote命令,会看到唯一的评审任务(编号1)有了两个补丁集。
$ git ls-remote origin 82c8fc3805d57cc0d17d58e1452e21428910fd2d HEAD c65ab490f6d3dc36429b8f1363b6191357202f2e refs/changes/01/1/1 1df9e8e05fcf97a46588488918a476abd1df8121 refs/changes/01/1/2 82c8fc3805d57cc0d17d58e1452e21428910fd2d refs/heads/master
5.6.10.6. 新修订集通过评审
当提交者重新针对评审任务进行提交时,原评审任务的审核者会收到通知邮件,提醒有新的补丁集等待评审,如图32-26。
图32-26:Gerrit通知邮件:新补丁集
登录Gerrit的Web界面,可以看到评审任务1有了新的补丁集,如图32-27。
图32-27:Gerrit新补丁集显示
再经过代码审核和测试,这次dev2
用户决定让评审通过,点击了“Publish and Submit”按钮。Submit(提交)动作会将评审任务(refs/changes/01/1/2)合并到对应分支(master)。图32-28显示的是通过评审完成合并的评审任务1。
图32-28:Gerrit合并后的评审任务
5.6.10.7. 从远程版本库更新
当dev1
和dev2
用户完成代码评审,提交者会收到多封通知邮件。这其中最让人激动的就是代码被接受并合并到开发主线(master)中(如图32-29),这是多么另开发者感到荣耀的啊。
图32-29:Gerrit通知邮件:修订已合并
代码提交者执行git pull,和Gerrit管理的版本库同步。
$ git ls-remote origin 1df9e8e05fcf97a46588488918a476abd1df8121 HEAD c65ab490f6d3dc36429b8f1363b6191357202f2e refs/changes/01/1/1 1df9e8e05fcf97a46588488918a476abd1df8121 refs/changes/01/1/2 1df9e8e05fcf97a46588488918a476abd1df8121 refs/heads/master $ git pull From ssh://localhost:29418/hello 82c8fc3..1df9e8e master -> origin/master Already up-to-date.
5.6.11. 更多Gerrit参考
Gerrit涉及到的内容非常庞杂,还有诸如和Gitweb、git-daemon整合,Gerrit界面定制等功能,恕不在此一一列举。可以直接参考Gerrit网站上的帮助。