控制測試如何運行
ch11-02-running-tests.md
commit 42b802f26197f9a066e4a671d2b062af25972c13
就像 cargo run
會編譯代碼並運行生成的二進位制文件一樣,cargo test
在測試模式下編譯代碼並運行生成的測試二進位制文件。可以指定命令行參數來改變 cargo test
的默認行為。例如,cargo test
生成的二進位制文件的默認行為是並行的運行所有測試,並截獲測試運行過程中產生的輸出,阻止他們被顯示出來,使得閱讀測試結果相關的內容變得更容易。
可以將一部分命令行參數傳遞給 cargo test
,而將另外一部分傳遞給生成的測試二進位制文件。為了分隔這兩種參數,需要先列出傳遞給 cargo test
的參數,接著是分隔符 --
,再之後是傳遞給測試二進位制文件的參數。運行 cargo test --help
會提示 cargo test
的有關參數,而運行 cargo test -- --help
可以提示在分隔符 --
之後使用的有關參數。
並行或連續的運行測試
當運行多個測試時, Rust 預設使用執行緒來並行運行。這意味著測試會更快地運行完畢,所以你可以更快的得到代碼能否工作的回饋。因為測試是在同時執行的,你應該確保測試不能相互依賴,或依賴任何共享的狀態,包括依賴共享的環境,比如當前工作目錄或者環境變數。
舉個例子,每一個測試都運行一些程式碼,假設這些程式碼都在硬碟上創建一個 test-output.txt 文件並寫入一些數據。接著每一個測試都讀取文件中的數據並斷言這個文件包含特定的值,而這個值在每個測試中都是不同的。因為所有測試都是同時執行的,一個測試可能會在另一個測試讀寫文件過程中修改了文件。那麼第二個測試就會失敗,並不是因為代碼不正確,而是因為測試並行運行時相互干擾。一個解決方案是使每一個測試讀寫不同的文件;另一個解決方案是一次執行一個測試。
如果你不希望測試並行運行,或者想要更加精確的控制執行緒的數量,可以傳遞 --test-threads
參數和希望使用執行緒的數量給測試二進位制文件。例如:
$ cargo test -- --test-threads=1
這裡將測試執行緒設置為 1
,告訴程序不要使用任何並行機制。這也會比並行運行花費更多時間,不過在有共享的狀態時,測試就不會潛在的相互干擾了。
顯示函數輸出
默認情況下,當測試通過時,Rust 的測試庫會截獲列印到標準輸出的所有內容。比如在測試中調用了 println!
而測試通過了,我們將不會在終端看到 println!
的輸出:只會看到說明測試通過的提示行。如果測試失敗了,則會看到所有標準輸出和其他錯誤訊息。
例如,範例 11-10 有一個無意義的函數,它列印出其參數的值並接著返回 10。接著還有一個會通過的測試和一個會失敗的測試:
檔案名: src/lib.rs
#![allow(unused)] fn main() { fn prints_and_returns_10(a: i32) -> i32 { println!("I got the value {}", a); 10 } #[cfg(test)] mod tests { use super::*; #[test] fn this_test_will_pass() { let value = prints_and_returns_10(4); assert_eq!(10, value); } #[test] fn this_test_will_fail() { let value = prints_and_returns_10(8); assert_eq!(5, value); } } }
運行 cargo test
將會看到這些測試的輸出:
running 2 tests
test tests::this_test_will_pass ... ok
test tests::this_test_will_fail ... FAILED
failures:
---- tests::this_test_will_fail stdout ----
I got the value 8
thread 'tests::this_test_will_fail' panicked at 'assertion failed: `(left == right)`
left: `5`,
right: `10`', src/lib.rs:19:9
note: Run with `RUST_BACKTRACE=1` for a backtrace.
failures:
tests::this_test_will_fail
test result: FAILED. 1 passed; 1 failed; 0 ignored; 0 measured; 0 filtered out
注意輸出中不會出現測試通過時列印的內容,即 I got the value 4
。因為當測試通過時,這些輸出會被截獲。失敗測試的輸出 I got the value 8
,則出現在輸出的測試摘要部分,同時也顯示了測試失敗的原因。
如果你希望也能看到通過的測試中列印的值,截獲輸出的行為可以通過 --nocapture
參數來禁用:
$ cargo test -- --nocapture
使用 --nocapture
參數再次運行範例 11-10 中的測試會顯示如下輸出:
running 2 tests
I got the value 4
I got the value 8
test tests::this_test_will_pass ... ok
thread 'tests::this_test_will_fail' panicked at 'assertion failed: `(left == right)`
left: `5`,
right: `10`', src/lib.rs:19:9
note: Run with `RUST_BACKTRACE=1` for a backtrace.
test tests::this_test_will_fail ... FAILED
failures:
failures:
tests::this_test_will_fail
test result: FAILED. 1 passed; 1 failed; 0 ignored; 0 measured; 0 filtered out
注意測試的輸出和測試結果的輸出是相互交叉的,這是由於測試是並行運行的(見上一部分)。嘗試一同使用 --test-threads=1
和 --nocapture
功能來看看輸出是什麼樣子!
透過指定名字來運行部分測試
有時運行整個測試集會耗費很長時間。如果你負責特定位置的代碼,你可能會希望只運行與這些程式碼相關的測試。你可以向 cargo test
傳遞所希望運行的測試名稱的參數來選擇運行哪些測試。
為了展示如何運行部分測試,範例 11-11 為 add_two
函數創建了三個測試,我們可以選擇具體運行哪一個:
檔案名: src/lib.rs
#![allow(unused)] fn main() { pub fn add_two(a: i32) -> i32 { a + 2 } #[cfg(test)] mod tests { use super::*; #[test] fn add_two_and_two() { assert_eq!(4, add_two(2)); } #[test] fn add_three_and_two() { assert_eq!(5, add_two(3)); } #[test] fn one_hundred() { assert_eq!(102, add_two(100)); } } }
如果沒有傳遞任何參數就運行測試,如你所見,所有測試都會並行運行:
running 3 tests
test tests::add_two_and_two ... ok
test tests::add_three_and_two ... ok
test tests::one_hundred ... ok
test result: ok. 3 passed; 0 failed; 0 ignored; 0 measured; 0 filtered out
運行單個測試
可以向 cargo test
傳遞任意測試的名稱來只運行這個測試:
$ cargo test one_hundred
Finished dev [unoptimized + debuginfo] target(s) in 0.0 secs
Running target/debug/deps/adder-06a75b4a1f2515e9
running 1 test
test tests::one_hundred ... ok
test result: ok. 1 passed; 0 failed; 0 ignored; 0 measured; 2 filtered out
只有名稱為 one_hundred
的測試被運行了;因為其餘兩個測試並不匹配這個名稱。測試輸出在摘要行的結尾顯示了 2 filtered out
表明還存在比本次所運行的測試更多的測試被過濾掉了。
不能像這樣指定多個測試名稱;只有傳遞給 cargo test
的第一個值才會被使用。不過有運行多個測試的方法。
過濾運行多個測試
我們可以指定部分測試的名稱,任何名稱匹配這個名稱的測試會被運行。例如,因為頭兩個測試的名稱包含 add
,可以通過 cargo test add
來運行這兩個測試:
$ cargo test add
Finished dev [unoptimized + debuginfo] target(s) in 0.0 secs
Running target/debug/deps/adder-06a75b4a1f2515e9
running 2 tests
test tests::add_two_and_two ... ok
test tests::add_three_and_two ... ok
test result: ok. 2 passed; 0 failed; 0 ignored; 0 measured; 1 filtered out
這運行了所有名字中帶有 add
的測試,也過濾掉了名為 one_hundred
的測試。同時注意測試所在的模組也是測試名稱的一部分,所以可以透過模組名來運行一個模組中的所有測試。
忽略某些測試
有時一些特定的測試執行起來是非常耗費時間的,所以在大多數運行 cargo test
的時候希望能排除他們。雖然可以透過參數列舉出所有希望運行的測試來做到,也可以使用 ignore
屬性來標記耗時的測試並排除他們,如下所示:
檔案名: src/lib.rs
#![allow(unused)] fn main() { #[test] fn it_works() { assert_eq!(2 + 2, 4); } #[test] #[ignore] fn expensive_test() { // 需要運行一個小時的代碼 } }
對於想要排除的測試,我們在 #[test]
之後增加了 #[ignore]
行。現在如果運行測試,就會發現 it_works
運行了,而 expensive_test
沒有運行:
$ cargo test
Compiling adder v0.1.0 (file:///projects/adder)
Finished dev [unoptimized + debuginfo] target(s) in 0.24 secs
Running target/debug/deps/adder-ce99bcc2479f4607
running 2 tests
test expensive_test ... ignored
test it_works ... ok
test result: ok. 1 passed; 0 failed; 1 ignored; 0 measured; 0 filtered out
expensive_test
被列為 ignored
,如果我們只希望運行被忽略的測試,可以使用 cargo test -- --ignored
:
$ cargo test -- --ignored
Finished dev [unoptimized + debuginfo] target(s) in 0.0 secs
Running target/debug/deps/adder-ce99bcc2479f4607
running 1 test
test expensive_test ... ok
test result: ok. 1 passed; 0 failed; 0 ignored; 0 measured; 1 filtered out
通過控制運行哪些測試,你可以確保能夠快速地運行 cargo test
。當你需要運行 ignored
的測試時,可以執行 cargo test -- --ignored
。