# 简介
在软件管理的领域里存在着被称作 依赖地狱
的死亡之谷,系统规模越大,加入的包越多,你就越有可能在未来的某一天发现自己已深陷绝望之中。
在依赖高的系统中发布新版本包可能很快会成为噩梦。如果依赖关系过高,可能面临版本控制被锁死的风险(必须对每一个依赖包改版才能完成某次升级)。而如果依赖关系过于松散,又将无法避免版本的混乱(假设兼容于未来的多个版本已超出了合理数量)。当你项目的进展因为版本依赖被锁死或版本混乱变得不够简便和可靠,就意味着你正处于依赖地狱之中。
作为这个问题的解决方案之一,我提议用一组简单的规则及条件来约束版本号的配置和增长。这些规则是根据(但不局限于)已经被各种封闭、开放源码软件所广泛使用的惯例所设计。为了让这套理论运作,你必须先有定义好的公共 API。这可能包括文档或代码的强制要求。无论如何,这套 API 的清楚明了是十分重要的。一旦你定义了公共 API,你就可以透过修改相应的版本号来向大家说明你的修改。考虑使用这样的版本号格式: 主版本号.次版本号.修订号
修复问题但不影响 API 时,递增修订号;API 保持向下兼容的新增及修改时,递增次版本号;进行不向下兼容的修改时,递增主版本号。
我称这套系统为 语义化的版本控制
,在这套约定下,版本号及其更新方式包含了相邻版本间的底层代码和修改内容的信息。
#语义化版本控制规范
以下关键词 MUST
、 MUST NOT
、 REQUIRED
、 SHALL
、 SHALL NOT
、 SHOULD
、 SHOULD NOT
、 RECOMMENDED
、 MAY
、 OPTIONAL
依照 RFC 2119 的叙述解读。
使用语义化版本控制的软件必须定义公共 API。该 API 可以在代码中被定义或出现于严谨的文档内。无论何种形式都应该力求精确且完整。
标准的版本号必须采用
X.Y.Z
的格式,其中 X、Y 和 Z 为非负的整数,且禁 止在数字前方补零。X 是主版本号、Y 是次版本号、而 Z 为修订号。每个元素必须以数值来递增。例如:1.9.1 -> 1.10.0 -> 1.11.0
。标记版本号的软件发行后,禁 止改变该版本软件的内容。任何修改都必须以新版本发行。
主版本号为零(0.y.z)的软件处于开发初始阶段,一切都可能随时被改变。这样的公共 API 不应该被视为稳定版。
1.0.0 的版本号用于界定公共 API 的形成。这一版本之后所有的版本号更新都基于公共 API 及其修改内容。
修订号 Z(x.y.Z | x > 0)必须在只做了向下兼容的修正时才递增。这里的修正指的是针对不正确结果而进行的内部修改。
次版本号 Y(x.Y.z | x > 0)必须在有向下兼容的新功能出现时递增。在任何公共 API 的功能被标记为弃用时也必须递增。也可以在内部程序有大量新功能或改进被加入时递增,其中可以包括修订级别的改变。每当次版本号递增时,修订号必须归零。
主版本号 X(X.y.z | X > 0)必须在有任何不兼容的修改被加入公共 API 时递增。其中可以包括次版本号及修订级别的改变。每当主版本号递增时,次版本号和修订号必须归零。
先行版本号可以被标注在修订版之后,先加上一个连接号再加上一连串以句点分隔的标识符来修饰。标识符必须由 ASCII 字母数字和连接号
[0-9A-Za-z-]
组成,且禁 止留白。数字型的标识符禁 止在前方补零。先行版的优先级低于相关联的标准版本。被标上先行版本号则表示这个版本并非稳定而且可能无法满足预期的兼容性需求。范例:1.0.0-alpha
、1.0.0-alpha.1
、1.0.0-0.3.7
、1.0.0-x.7.z.92
。版本编译信息可以被标注在修订版或先行版本号之后,先加上一个加号再加上一连串以句点分隔的标识符来修饰。标识符必须由 ASCII 字母数字和连接号
[0-9A-Za-z-]
组成,且禁 止留白。当判断版本的优先层级时,版本编译信息可被忽略。因此当两个版本只有在版本编译信息有差别时,属于相同的优先层级。范例:1.0.0-alpha+001
、1.0.0+20130313144700
、1.0.0-beta+exp.sha.5114f85
。版本的优先层级指的是不同版本在排序时如何比较。
判断优先层级时,必须(MUST)把版本依序拆分为主版本号、次版本号、修订号及先行版本号后进行比较(版本编译信息不在这份比较的列表中)。
由左到右依序比较每个标识符,第一个差异值用来决定优先层级:主版本号、次版本号及修订号以数值比较。
例如:
1.0.0
<2.0.0
<2.1.0
<2.1.1
。当主版本号、次版本号及修订号都相同时,改以优先层级比较低的先行版本号决定。
例如:
1.0.0-alpha
<1.0.0
。有相同主版本号、次版本号及修订号的两个先行版本号,其优先层级必须(MUST)透过由左到右的每个被句点分隔的标识符来比较,直到找到一个差异值后决定:
- 只有数字的标识符以数值高低比较。
- 有字母或连接号时则逐字以 ASCII 的排序来比较。
- 数字的标识符比非数字的标识符优先层级低。
- 若开头的标识符都相同时,栏位比较多的先行版本号优先层级比较高。
例如:
1.0.0-alpha
<1.0.0-alpha.1
<1.0.0-alpha.beta
<1.0.0-beta
<1.0.0-beta.2
<1.0.0-beta.11
<1.0.0-rc.1
<1.0.0
。
# 合法语义化版本的巴科斯范式语法
<valid semver> ::= <version core> | |
| <version core> "-" <pre-release> | |
| <version core> "+" <build> | |
| <version core> "-" <pre-release> "+" <build> | |
<version core> ::= <major> "." <minor> "." <patch> | |
<major> ::= <numeric identifier> | |
<minor> ::= <numeric identifier> | |
<patch> ::= <numeric identifier> | |
<pre-release> ::= <dot-separated pre-release identifiers> | |
<dot-separated pre-release identifiers> ::= <pre-release identifier> | |
| <pre-release identifier> "." <dot-separated pre-release identifiers> | |
<build> ::= <dot-separated build identifiers> | |
<dot-separated build identifiers> ::= <build identifier> | |
| <build identifier> "." <dot-separated build identifiers> | |
<pre-release identifier> ::= <alphanumeric identifier> | |
| <numeric identifier> | |
<build identifier> ::= <alphanumeric identifier> | |
| <digits> | |
<alphanumeric identifier> ::= <non-digit> | |
| <non-digit> <identifier characters> | |
| <identifier characters> <non-digit> | |
| <identifier characters> <non-digit> <identifier characters> | |
<numeric identifier> ::= "0" | |
| <positive digit> | |
| <positive digit> <digits> | |
<identifier characters> ::= <identifier character> | |
| <identifier character> <identifier characters> | |
<identifier character> ::= <digit> | |
| <non-digit> | |
<non-digit> ::= <letter> | |
| "-" | |
<digits> ::= <digit> | |
| <digit> <digits> | |
<digit> ::= "0" | |
| <positive digit> | |
<positive digit> ::= "1" | "2" | "3" | "4" | "5" | "6" | "7" | "8" | "9" | |
<letter> ::= "A" | "B" | "C" | "D" | "E" | "F" | "G" | "H" | "I" | "J" | |
| "K" | "L" | "M" | "N" | "O" | "P" | "Q" | "R" | "S" | "T" | |
| "U" | "V" | "W" | "X" | "Y" | "Z" | "a" | "b" | "c" | "d" | |
| "e" | "f" | "g" | "h" | "i" | "j" | "k" | "l" | "m" | "n" | |
| "o" | "p" | "q" | "r" | "s" | "t" | "u" | "v" | "w" | "x" | |
| "y" | "z" |
# 为什么要使用语义化的版本控制?
这并不是一个新的或者革命性的想法。实际上,你可能已经在做一些近似的事情了。问题在于只是 “近似” 还不够。如果没有某个正式的规范可循,版本号对于依赖的管理并无实质意义。将上述的想法命名并给予清楚的定义,让你对软件使用者传达意向变得容易。一旦这些意向变得清楚,弹性(但又不会太弹性)的依赖规范就能达成。
举个简单的例子就可以展示语义化的版本控制如何让依赖地狱成为过去。假设有个名为 “救火车” 的函数库,它需要另一个名为 “梯子” 并已经有使用语义化版本控制的包。当救火车创建时,梯子的版本号为 3.1.0。因为救火车使用了一些版本 3.1.0 所新增的功能,你可以放心地指定依赖于梯子的版本号大于等于 3.1.0 但小于 4.0.0。这样,当梯子版本 3.1.1 和 3.2.0 发布时,你可以将直接它们纳入你的包管理系统,因为它们能与原有依赖的软件兼容。
作为一位负责任的开发者,你理当确保每次包升级的运作与版本号的表述一致。现实世界是复杂的,我们除了提高警觉外能做的不多。你所能做的就是让语义化的版本控制为你提供一个健全的方式来发行以及升级包,而无需推出新的依赖包,节省你的时间及烦恼。
如果你对此认同,希望立即开始使用语义化版本控制,你只需声明你的函数库正在使用它并遵循这些规则就可以了。请在你的 README
文件中保留此页链接,让别人也知道这些规则并从中受益。