提交代码¶
本部分针对提交者以及任何有兴趣了解如何为 Django 贡献代码的人员。 如果您是想向 Django 贡献代码的社区成员,请查看 使用 Git 和 GitHub 工作
处理 pull 请求¶
因为Django现在是托管在GitHub上的,所以大多数补丁都是以 pull requests 的形式提供的。
提交拉取请求时,请确保每个单独的提交符合下面描述的提交准则。我们期望贡献者提供最好的拉取请求。在实践中,合并者(很可能更熟悉提交准则)可能会决定自行将提交符合标准。
你可能希望使用 Jenkins 或 GitHub actions 来测试拉取请求,其中之一是不会自动运行的拉取请求构建器,比如 Oracle 或 Selenium。请查看 CI wiki 页面 获取说明。
如果您发现自己经常在本地检出拉取请求,这个 Git 别名将会很有帮助:
[alias]
pr = !sh -c \"git fetch upstream pull/${1}/head:pr/${1} && git checkout pr/${1}\"
将它添加到你的 ~/.gitconfig
文件中,并且把 upstream
改为 django/django
。然后你就可以通过运行 git pr ####
来检出对应的 pull request 了。
到了这个阶段,你可以着手于代码.使用"git rebase -i"和"git commit --ammend" 来确保你的提交准确无误. 一旦你准备好了:
$ # Pull in the latest changes from main.
$ git checkout main
$ git pull upstream main
$ # Rebase the pull request on main.
$ git checkout pr/####
$ git rebase main
$ git checkout main
$ # Merge the work as "fast-forward" to main to avoid a merge commit.
$ # (in practice, you can omit "--ff-only" since you just rebased)
$ git merge --ff-only pr/XXXX
$ # If you're not sure if you did things correctly, check that only the
$ # changes you expect will be pushed to upstream.
$ git push --dry-run upstream main
$ # Push!
$ git push upstream main
$ # Delete the pull request branch.
$ git branch -d pr/xxxx
...\> REM Pull in the latest changes from main.
...\> git checkout main
...\> git pull upstream main
...\> REM Rebase the pull request on main.
...\> git checkout pr/####
...\> git rebase main
...\> git checkout main
...\> REM Merge the work as "fast-forward" to main to avoid a merge commit.
...\> REM (in practice, you can omit "--ff-only" since you just rebased)
...\> git merge --ff-only pr/XXXX
...\> REM If you're not sure if you did things correctly, check that only the
...\> REM changes you expect will be pushed to upstream.
...\> git push --dry-run upstream main
...\> REM Push!
...\> git push upstream main
...\> REM Delete the pull request branch.
...\> git branch -d pr/xxxx
在将主分支重新基于重置后、合并并推送到上游之前,强制推送到分支。这样可以使主分支和分支上的提交哈希匹配,自动关闭拉取请求。
如果拉请求不需要合并为多个提交,则可以在网页上使用 Github 的“Squash and merge”按钮。根据需要编辑提交消息以符合 指南 并删除自动附加到消息第一行的请求编号。
当重写pull请求的提交历史时,目标是使Django的提交历史尽可能可用:
如果补丁包含来回提交,则将其重写为一个。例如,如果提交添加了一些代码,第二个提交修复了第一次提交中引入的样式问题,那么这些提交应该在合并之前被压缩。
通过逻辑分组对不同提交进行单独更改:如果在对文件进行其他更改的同时进行样式清理,则将更改分成两个不同的提交将使查看历史记录变得更容易。
注意拉取请求中上游分支的合并。
测试应该通过,文档应该在每次提交后构建。测试和文档都不应该发出警告。
随意和小型的补丁通常最好在一次提交中完成。如果有意义,可以将中型到大型工作分成多个提交。
实用性胜过纯粹性,因此由每个合并者决定拉取请求的历史重塑程度。主要目标是与社区互动、完成工作并拥有可用的提交历史。
提交指南¶
此外,在将代码提交到Django的Git存储库时,请遵循以下准则:
永远不要通过强制推动来改变“django / django``分支的已发布历史。如果您绝对必须(出于安全原因),请先与团队讨论情况。
对于任何中等到大规模的更改,其中 "中等到大规模" 根据你的判断,都请在进行更改之前在 Django 论坛 或 django-developers 邮件列表上提出讨论。
如果你提出了某个问题,而没有人回应,请不要误以为你的想法很好,并且应该立即实施,因为没有人反对它。并不是每个人都有足够的时间立即阅读邮件列表中的讨论,所以你可能需要等待几天才能得到回应。
以过去时态写出详细的提交消息,而不是现在时。
正确:"Fixed Unicode bug in RSS API."
错误:"Fixes Unicode bug in RSS API."
错误:"Fixing Unicode bug in RSS API."
提交消息最多应为72个字符。 应该有一个主题行,用空白行分隔,然后是72个字符行的段落。 限制很软。 对于主题行,越短越好。 在提交消息的主体中,更多细节优于更少:
Fixed #18307 -- Added git workflow guidelines. Refactored the Django's documentation to remove mentions of SVN specific tasks. Added guidelines of how to use Git, GitHub, and how to use pull request together with Trac instead.
可以在提交消息中添加对贡献者的感谢:“感谢 A 提供报告,B 提供补丁,C 提供审查。” 也推荐使用 git 的 Co-Authored-By 功能。
对于提交到分支,请在提交消息前面加上分支名称。例如:“[1.4.x]Fixed XXXX——增加了对思维阅读的支持。”
限制承诺最细微的变化是有意义的。这意味着,使用频繁的小额承诺,而不是很少的大额承诺。例如,如果实现功能X需要对库Y做一个小的更改,那么首先将更改提交到库Y,然后在单独的提交中提交功能X。这对帮助每个人跟踪你的变化有很大的帮助。
将错误修复与功能更改分开。根据 支持的版本,可能需要将错误修复返回到稳定分支。
如果你的提交解决了 Django ticket tracker 中的一个 ticket,请在提交信息的开头添加 "Fixed #xxxxx",其中 ”xxxxx" 是你提交所解决的 ticket 的编号。示例:"Fixed #123 -- Added whizbang feature." 。当系统检测到这种提交信息格式时,系统会自动关闭所指向的 ticket 并将完整的提交信息作为评论发出。
如果你好奇的话,我们使用 Trac plugin 来实现这种机制。
备注
请注意,Trac 集成对拉取请求一无所知。因此,如果你在提交信息中使用了 " closes #400 " 这样的短语来关闭拉取请求,GitHub 将关闭该拉取请求,但 Trac 插件将不会关闭 Trac 中相同编号的工单。
如果你只想要在提交信息中引用 Django ticket tracker 中的某个 ticket ,但并*不*想要关闭它时,可以使用 “Ref #xxxxx",其中 "xxxxx" 是你的提交想引用的 ticket 的编号。这会触发系统自动的在被引用的 ticket 下发送一条评论。
使用此模式为后端端口编写提交消息:
[<Django version>] Fixed <ticket> -- <description> Backport of <revision> from <branch>.
例如:
[1.3.x] Fixed #17028 -- Changed diveintopython.org -> diveintopython.net. Backport of 80c0cbf1c97047daed2c5b41b296bbc56fe1d7e3 from main.
有一个 维基上的脚本 可以自动化这个过程。
如果提交修复了回归,请将其包含在提交消息中:
Regression in 6ecccad711b52f9273b1acb07a57d3f806e93928.
(使用引入回归的提交哈希)。
恢复提交¶
没有人是完美的;一些错误可能会被提交
但要尽力确保不出现错误。仅仅因为我们有一个回退政策并不意味着你的责任就可以放松,你仍然需要追求最高的质量。真的:在首次提交之前,仔细检查你的工作,或者让另一个合并者在你提交之前检查它!
当你发现了一个错误的提交,请遵循如下步骤:
可能的话,让原始作者撤回他们的提交。
不要在未获得原始作者同意的情况下还原其他作者的更改。
使用 git revert -- 会回退提交,但原提交依然会留在提交历史中。
如果无法联系到原作者(在合理的时间内,大约一天左右),而问题很严重,比如严重的错误、重大的测试失败等等,那么请在 Django 论坛 或 django-developers 邮件列表上征求意见,如果没有反对意见,那么就回退更改。
如果问题不大(比如说冻结后的提交),就等着吧。
如果在合并者和即将回退更改的人之间存在分歧,那么请尝试在 Django 论坛 或 django-developers 邮件列表上解决。如果无法达成一致意见,那么应该进行投票。
如果该提交引入了一个已确认或已披露的安全漏洞,那么该提交可能会在未经任何允许的情况下被立即还原。
如果提交的内容破坏了发布分支,发布分支的维护者可以在不经许可的情况下将该提交回退。
如果错误地将主题分支推送到 "django/django",请将其删除。例如,如果你做了:
git push upstream feature_antigravity`
,那么做一个反向推送:git push upstream:feature_antigravity
。