Rust:带有 lib 和二进制目标的板条箱中的属性 #[cfg(test)]

Chr*_*uhl 2 testing conditional-compilation rust rust-cargo

我目前正在研究一个Rust 项目,该项目有一个库目标和一个二进制目标。

该库定义了两种恐慌处理程序,一种用于生产使用,另一种用于测试,它们使用 #[cfg(test)] 属性进行条件编译。这对于单元测试 src/lib.rs 来说效果很好,但是一旦在tests/(或 src/main.rs)中进行集成测试,“生产”恐慌处理程序就会被链接到,即在这些情况下,lib。 rs 使用 test=false 进行编译。

有什么方法可以防止/配置它吗?

使用以下目录结构可以最低程度地重现这一点:

Cargo.toml
src/lib.rs
src/main.rs
Run Code Online (Sandbox Code Playgroud)

清单如下所示:

# Cargo.toml
...
[lib]
name = "kernel"

[[bin]]
path = "src/main.rs"
name = "titanium"
Run Code Online (Sandbox Code Playgroud)

二进制代码:

// main.rs

fn main() {
    println!("Hello, world!");
}

#[test]
fn it_works() {
    assert_eq!(kernel::hello(), "Hello!".to_string());
}
Run Code Online (Sandbox Code Playgroud)

...对于图书馆:

// lib.rs

#[cfg(test)]
pub fn hello() -> String {
    "Hello!".to_string()
}

#[cfg(not(test))]
pub fn hello() -> String {
    "Goodbye!".to_string()
}

#[test]
fn it_works() {
    assert_eq!(hello(), "Hello!".to_string())
}
Run Code Online (Sandbox Code Playgroud)

回答 事实上,上述行为对于大约 99% 的用例来说都是可取的,但是,我的不是其中之一。

对于我的用例(在“无头”虚拟机中运行的玩具操作系统的测试套件,即仅输出到主机的标准输出),我最终定义了一个功能来切换恐慌处理程序。适应上面的例子,现在看起来像这样:

# Cargo.toml
...
[features]
test_qemu_headless = []
Run Code Online (Sandbox Code Playgroud)

然后二进制文件保持不变,并且在库中我做

// lib.rs

#[cfg(feature = "test_qemu_headless")]
pub fn hello() -> String {
    "Hello!".to_string()
}

#[cfg(not(feature = "test_qemu_headless"))]
pub fn hello() -> String {
    "Goodbye!".to_string()
}

#[test]
fn it_works() {
    assert_eq!(hello(), "Hello!".to_string())
}
Run Code Online (Sandbox Code Playgroud)

然后运行测试

cargo test --features test_qemu_headless
Run Code Online (Sandbox Code Playgroud)

use*_*968 5

lib 目标被编译两次:一次是在运行单元测试时;一次是在运行单元测试时;一次是在运行单元测试时。此处,#[cfg(test)]处于活动状态,因为该库当前正在编译为rustc --test. 第二次运行集成测试时;此处,#[cfg(test)]未激活因为该库正在正常编译,然后链接到集成测试模块,这些模块本身被编译为测试。换句话说,当与集成测试一起使用时,lib 目标不知道它正在被编译并在测试中使用(这在某种程度上是集成测试的重点)。

您正在寻找的是#[cfg(debug_assertions)]装饰您的恐慌处理程序。无论出于何种原因,只要代码在“调试”模式下编译,此属性就会处于活动状态。使用此属性,您的测试处理程序将对单元测试和集成测试都有效。

但是,如果您在“调试”模式下编译 bin 目标并运行它(例如通过cargo runwithout --release),或者如果您的库依赖于在“调试”模式下编译的其他板条箱,则库中的测试处理程序将也要积极主动,但这可能不是您想要的。如果是这样,您需要feature为您的板条箱定义一个 ,依赖于您的恐慌处理程序的测试可以将其用作条件编译的标志(例如foocrate_use_internal_dummy_panic_handler)。然后,您需要在运行测试时激活该功能 - 这将成为全局旋钮。required-feature清单中还有一个- 字段。