您的箱子,可以依赖多个来源的库,如crates.io,git
的存储库或本地文件系统上的子目录。您还可以临时覆盖依赖项的位置 - 例如, 便于能够测试您在本地工作的依赖项中的错误修复。您可以为不同的平台,和或仅在开发期间使用不同的依赖项。我们来看看如何做到这些.
默认情况下,Cargo 是准备好,在crates.io上查找依赖项。在这种情况下,只需要名称和版本字符串。在Cargo 指南,我们选择了一个依赖项-time
箱:
[dependencies]
time = "0.1.12"
字符串"0.1.12"
是一个semver版本格式字符串。由于此字符串中没有任何运算符,因此它的解释方式与我们指定的"^0.1.12"
方式相同,而^
被称为跳脱条件.
跳脱条件: 允许 SemVer 兼容更新指定版本。新的版本允许更新的条件是,不修改最左边的非零数字(无论major,minor,patch
)。在这种情况下,如果我们执行了cargo update -p time
,Cargo 应该更新我们的0.1.13
版本(如果是最新的0.1.z
发布),但不会更新为0.2.0
。相反,我们若将版本字符串指定为^1.0
,Cargo 应更新至1.1
,如果是最新的1.y
发布,但不是2.0
版本。0.0.x
并不与任何其他版本兼容.
以下是一些跳脱条件的例子以及它们允许的版本:
^1.2.3 := >=1.2.3 <2.0.0
^1.2 := >=1.2.0 <2.0.0
^1 := >=1.0.0 <2.0.0
^0.2.3 := >=0.2.3 <0.3.0
^0.2 := >= 0.2.0 < 0.3.0
^0.0.3 := >=0.0.3 <0.0.4
^0.0 := >=0.0.0 <0.1.0
^0 := >=0.0.0 <1.0.0
此兼容性约定与 SemVer ,在处理 1.0.0 之前的版本方面有所不同。虽然 SemVer 说在 1.0.0 之前没有兼容性,但 Cargo 认为0.x.y
是兼容0.x.z
,这里y ≥ z
和x > 0
.
Tilde 条件指定具有更新最小版本的一定能力。如果指定 major 版本,minor 版本和 patch 程序版本,或仅指定 major 版本和 minor 版本,则仅允许 patch 程序级别更改。如果仅指定 major 版本,则允许进行 minor 和 patch 级别更改.
~1.2.3
是 Tilde 条件的一个例子.
~1.2.3 := >=1.2.3 <1.3.0
~1.2 := >=1.2.0 <1.3.0
~1 := >=1.0.0 <2.0.0
通配符条件允许任何通配符所在的版本.
*
,1.*
和1.2.*
是通配符条件的示例.
* := >=0.0.0
1.* := >=1.0.0 <2.0.0
1.2.* := >=1.2.0 <1.3.0
范围条件允许手动指定要依赖的版本范围或确切版本.
以下是范围条件的一些示例:
>= 1.2.0
> 1
< 2
= 1.2.3
多个版本,要求用逗号分隔,例如>= 1.2, < 1.5
.
依赖于位于git
存储库的库,您需要指定的最小信息,为一个git
字段,其是存储库的github
位置:
[dependencies]
rand = { git = "https://github.com/rust-lang-nursery/rand" }
Cargo 将取得git
,然后在这个位置找到一个存储库的请求箱子的Cargo.toml
。方式是对git
存储库里面的任何地方(不一定在根目录) - 例如,指定工作区中的成员包名称,和设置git
到包含工作区的存储库).
由于我们尚未指定任何其他信息,因此 Cargo 假定我们打算使用最新的提交master
分支,来构建我们的包。你可以将git
字段和rev
,tag
, 还有branch
,这些用于指定其他内容的字段组合起来。这是一个指定您希望在名为next
分支上,使用最新提交的示例:
[dependencies]
rand = { git = "https://github.com/rust-lang-nursery/rand", branch = "next" }
随着时间的推移,我们来自指南的hello_world
示例已大幅增长! 它已经到了我们可能想分出一个单独的箱子供其他人使用的地步。为此,Cargo 支持路径依赖通常是位于一个存储库中的子箱。让我们开始在hello_world
包的内部制作一个新的箱子:
# inside of hello_world/
$ cargo new hello_utils
这将创建一个新文件夹hello_utils
,里面有一个Cargo.toml
和src
文件夹已准备好进行配置。为了告诉 Cargo,请打开hello_world/Cargo.toml
,并添加你的hello_utils
依赖:
[dependencies]
hello_utils = { path = "hello_utils" }
这告诉 Cargo 我们依赖于一个叫做hello_utils
的箱子,这能在hello_utils
文件夹找到(相对于,写在Cargo.toml
路径).
就是这样! 下一步cargo build
将自动构建hello_utils
,以及它自己的所有依赖项,其他人也可以开始使用它。但是,crates.io不允许仅使用 路径指定依赖项 的包。如果我们想发布我们的hello_world
箱子,我们需要发布一个版本hello_utils
至crates.io,并在依赖项行中指定其版本:
[dependencies]
hello_utils = { path = "hello_utils", version = "0.1.0" }
Cargo 中有许多方法支持,覆盖依赖关系以及控制依赖关系图。但是,这些选项通常仅在工作区级别可用,并且不通过依赖项传播。换句话说,"应用程序"具有覆盖依赖关系的能力,但"库"却没有。
许多场景,会产生想,覆盖依赖性或以其他方式改变某些依赖关系的愿望。然而,他们中的大多数都可以归结为,将箱子发布到 crates.io 之前使用箱子(覆盖依赖)的能力。例如:
- 您编写的
crate
,也用于您编写的更大应用程序中,并且您希望测试在更大应用程序内,crate
的错误修复情况。 - 不是你编写的上游包,现在其 git 存储库的主分支上,有一个新功能或错误修复,您要测试它。
- 您即将发布新版本的 major 版本,但您希望在整个软件包中进行集成测试,以确保新的主要版本能够正常运行.
- 您已经为上游的软件包提交了一个针对您找到的错误的修复程序,但是您希望立即让您的应用程序依赖,此程序包的固定修复版本,以避免错误修复程序被拒绝合并.
这些场景目前都是通过[patch]
清单部分 解决的,从历史上看,其中一些方案是该[replace]
部分解决的,但我们在这里会记录[patch]
解决的部分。
假设你正在使用uuid
crate,但是当你正在研究它时,你会发现一个错误.但是,你很有进取心,所以你决定尝试修复这个 bug! 最初你的清单看起来像:
[package]
name = "my-library"
version = "0.1.0"
authors = ["..."]
[dependencies]
uuid = "1.0"
我们要做的第一件事是克隆uuid
存储库,到本地:
$ git clone https://github.com/rust-lang-nursery/uuid
接下来我们将编辑my-library
-Cargo.toml,为:
[patch.crates-io]
uuid = { path = "../path/to/uuid" }
在这里,我们宣布我们是*修补(patch)*来源crates-io
,其有一个新的依赖,这将有效地添加本地(签出 checkout)版本uuid
到 crates.io 注册表,指向本地包。
接下来我们需要确保我们的锁(lock)文件已更新为,使用此新版本uuid
,所以我们的包使用本地签出的副本,而不是 crates.io 中的副本。[patch]
工作方式是它将从../path/to/uuid
加载依赖,然后每当 crates.io 查询uuid
的版本时,它也会返回本地版本.
这意味着本地签出的版本号很重要,会影响是否使用该补丁。我们的清单宣布uuid = "1.0"
,这意味着我们只会解析>= 1.0.0, < 2.0.0
,和 Cargo 的贪婪解析算法,也意味着我们将解析到该范围内的最大版本。通常情况下这并不重要,因为 git 存储库的版本已经更大,或与 crates.io 上发布的最大版本相匹配,但重要的是要记住这一点!
无论如何,通常您现在需要做的就是:
$ cargo build
Compiling uuid v1.0.0 (.../uuid)
Compiling my-library v0.1.0 (.../my-library)
Finished dev [unoptimized + debuginfo] target(s) in 0.32 secs
就是这样! 您现在正在使用本地版本uuid
构建(注意构建输出中括号中的路径)。如果您没有看到构建本地路径版本,那么您可能需要运行cargo update -p uuid --precise $version
,这里$version
是本地签出版本的uuid
副本。
一旦你修复了你最初发现的错误,你要做的下一件事就是将其作为拉取请求提交给uuid
箱子本身。一旦你完成了这个,你也可以更新下[patch]
部分。[patch]
里面的内容列表就像是[dependencies]
部分,所以一旦你的拉动请求合并,你就可以改变你的path
依赖:
[patch.crates-io]
uuid = { git = 'https://github.com/rust-lang-nursery/uuid' }
与 一个未发布的次要版本,一起工作
现在让我们稍微改变一下,从错误修复,变成要添加功能。在努力my-library
的同时,你发现需要uuid
箱的一个全新的功能。而您已实现uuid
此功能,并在[patch]
上面进行本地测试,并提交了拉取请求。让我们来看看在实际发布之前,你如何继续使用和测试它。
我们也说当前版本的uuid
,在 crates.io 上是1.0.0
版本,但从提交那时起,git 存储库的主分支已更新为1.0.1
。此分支包含您之前提交的新功能。要使用此存储库,我们将编辑我们的Cargo.toml
,看起来像
[package]
name = "my-library"
version = "0.1.0"
authors = ["..."]
[dependencies]
uuid = "1.0.1"
[patch.crates-io]
uuid = { git = 'https://github.com/rust-lang-nursery/uuid' }
注意我们对本地uuid
的依赖已更新为1.0.1
,因为这是我们在箱子发布后实际需要的东西。但是,这个版本在 crates.io 上不存在,所以我们提供给它清单的[patch]
部分.
现在,当我们的库被构建时,它将uuid
从 git 存储库取出,并解析到存储库中的 1.0.1 ,而不是尝试从 crates.io 下载版本。一旦 1.0.1 发布在 crates.io 上,那[patch]
部分就可以删除了。
值得注意的是,[patch]
是连带关系。假设您在更大的包中使用my-library
,例如:
[package]
name = "my-binary"
version = "0.1.0"
authors = ["..."]
[dependencies]
my-library = { git = 'https://example.com/git/my-library' }
uuid = "1.0"
[patch.crates-io]
uuid = { git = 'https://github.com/rust-lang-nursery/uuid' }
记住这[patch]
是连带关系,但只能在顶层,所以我们的my-library
消费者不得不重写[patch]
部分(如有必要的话)。不过,在这里,新的uuid
箱子会适用对uuid
的依赖和my-library -> uuid
的依赖,两个依赖都指定了。该uuid
箱 将被解析为整个 crate 关系图 的 1.0.1 版本,并且它是将从 git 存储库中提取。
覆盖 注册表 URL
如果要覆盖的依赖项不是加载自crates.io
,你将不得不改变一下你的[patch]
使用方式:
[patch."https://github.com/your/repository"]
my-library = { path = "../my-library/path" }
就是这样!
预发布一个重要变化
让我们来看看最后一个场景。若要使用一个新的主要版本的箱子,其通常伴随着重大变化。而要坚持使用我们以前的箱,这意味着我们将创建 2.0.0 版本uuid
箱。在我们提交了所有上游更改后,我们可以更新我们的my-library
清单,看起来像:
[dependencies]
uuid = "2.0"
[patch.crates-io]
uuid = { git = "https://github.com/rust-lang-nursery/uuid", branch = "2.0.0" }
就是这样!与前面的示例一样,2.0.0 版本实际上,并不存在于 crates.io 上,但我们仍然可以通过[patch]
部分使用。作为一个思考练习,让我们再看看my-binary
(被使用)的再次表现:
[package]
name = "my-binary"
version = "0.1.0"
authors = ["..."]
[dependencies]
my-library = { git = 'https://example.com/git/my-library' }
uuid = "1.0"
[patch.crates-io]
uuid = { git = 'https://github.com/rust-lang-nursery/uuid', branch = '2.0.0' }
请注意,这实际上将解析为两个版本的uuid
箱。该my-binary
箱子将继续使用 1.x.y 系列的uuid
箱子,但是my-library
箱 会使用 2.0.0 版本uuid
。这将允许您通过依赖关系图逐步推出对包的更改,而无需一次性更新所有内容。
覆盖 本地依赖项
有时你只是暂时在一个箱子上工作,而你不想修改Cargo.toml
中像上诉的[patch]
部分。对于这个用例,Cargo 提供了更为有限的覆盖版本路径覆盖.
路径覆盖是通过.cargo/config
指定,而不是Cargo.toml
,你可以寻找有关此配置的更多文档。在.cargo/config
内,你要指定的是一个名为paths
字段:
paths = ["/path/to/uuid"]
该数组应填充包含Cargo.toml
的目录。在这种情况下,我们只是添加uuid
,所以它将是唯一一个被覆盖的。此路径可以是包含该路径的绝对路径或相对.cargo
文件夹的路径.
路径覆盖,比[patch]
部分的限制更严格,但是,路径覆盖不能改变依赖图的结构。而当使用路径替换时,前一组依赖项必须完全匹配新的Cargo.toml
规格。如此,就意味着路径覆盖不能用于向箱添加依赖项的测试,而换成[patch]
在该种情况下使用。因此,路径覆盖的使用,通常会与快速错误修复分隔开来,而不是大更新分开。
注意:使用本地配置覆盖路径,仅适用于已发布到crates.io的包。您无法使用此功能告诉 Cargo 如何查找本地未发布的箱。
平台决定依赖
特定于平台的依赖项采用相同的格式,但在target
下列出。像正常 Rust 一样的#[cfg]
语法,将用于定义这些部分:
[target.'cfg(windows)'.dependencies]
winhttp = "0.4.0"
[target.'cfg(unix)'.dependencies]
openssl = "1.0.1"
[target.'cfg(target_arch = "x86")'.dependencies]
native = { path = "native/i686" }
[target.'cfg(target_arch = "x86_64")'.dependencies]
native = { path = "native/x86_64" }
与 Rust 一样,这里的语法支持not
,any
,和all
运算符组合各种 cfg 名称/值对。请注意cfg
语法仅在 Cargo 0.9.0(Rust 1.8.0)之后可用.
除了#[cfg]
语法,Cargo 还支持列出依赖关系适用的完整目标:
[target.x86_64-pc-windows-gnu.dependencies]
winhttp = "0.4.0"
[target.i686-unknown-linux-gnu.dependencies]
openssl = "1.0.1"
如果您使用的是自定义目标规范,请引用完整路径和文件名:
[target."x86_64/windows.json".dependencies]
winhttp = "0.4.0"
[target."i686/linux.json".dependencies]
openssl = "1.0.1"
native = { path = "native/i686" }
[target."x86_64/linux.json".dependencies]
openssl = "1.0.1"
native = { path = "native/x86_64" }
开发(Dev)依赖项
你可以添加一个[dev-dependencies]
表格到Cargo.toml
,其格式相当于[dependencies]
:
[dev-dependencies]
tempdir = "0.3"
编译用于构建的包时,不会使用 Dev 依赖,但用于编译测试,示例和基准。
这些依赖关系是不会传播到依赖于此包的其他包.
您还可以让dev-dependencies
具有特定目标的开发依赖项,而不是dependencies
标题。例如:
[target.'cfg(unix)'.dev-dependencies]
mio = "0.0.1"
构建 依赖项
您可以在构建脚本中使用,依赖其他基于 Cargo 的箱。依赖关系是由清单的build-dependencies
部分定义:
[build-dependencies]
cc = "1.0.3"
构建脚本并不是有权访问中 dependencies要么
dev-dependencies部分列出的依赖项
。除非也在dependencies
部分下面列出,否则构建依赖项同样不可用于包本身。包本身及其构建脚本是分开构建的,因此它们的依赖关系不重合。通过将独立依赖用于独立目的,使 Cargo 更简单,更清洁。
选择 特性
如果您依赖的包提供条件特性,您可以指定使用哪个:
[dependencies.awesome]
version = "1.3.5"
default-features = false # 不会包括默认特性, 和 任君选
# 单特性
features = ["secure-password", "civet"]
有关 features 的更多信息,请参阅清单文档.
在
Cargo.toml
中的重命名依赖项
写Cargo.toml
的[dependencies]
部分的时候,您为依赖项编写的字段通常与您在代码中导入的包的名称相匹配。但是,对于某些项目,您可能希望在代码中引用具有不同名称的包,而不管它是如何在 crates.io 上发布的。例如,您可能希望:
- 避免在 Rust 代码常用
use foo as bar
. - 依赖箱子的多个版本.
- 依赖来自不同注册表管理机构的同名箱.
为了支持这个 ,Cargo 在[dependencies]
部分使用 一个package
字段,决定应该依赖哪个包:
[package]
name = "mypackage"
version = "0.0.1"
[dependencies]
foo = "0.1"
bar = { git = "https://github.com/example/project", package = "foo" }
baz = { version = "0.1", registry = "custom", package = "foo" }
在此示例中,Rust 代码中现在提供了三个包:
extern crate foo; // crates.io
extern crate bar; // git repository
extern crate baz; // registry `custom`
所有这三个箱的包名称在他们自己Cargo.toml
,都是foo
,所以我们明确地告知 Cargo ,使用的是我们想要的package
字段(如 package = "foo"包名,即我们在本地调用其他东西)。如果没有指定package
,则默认为所请求的依赖项的名称。
请注意,如果您有一个可选的(optional)依赖项,例如:
[dependencies]
foo = { version = "0.1", package = 'bar', optional = true }
你依赖于一个bar
箱子,其来自 crates.io,但你箱子有一个foo
特性,取代了一个bar
特性。也就是说,在重命名时,特性的名称拿掉了依赖项的名称,而不是包名称。
启用传递依赖项的工作方式类似,例如我们可以将以下内容,添加到上面的清单中:
[features]
log-debug = ['foo/log-debug'] # 使用 'bar/log-debug' 就会出现一个错误!