<style> .textleft { text-align:left; } .reveal, .reveal h1, .reveal h2, .reveal h3, .reveal h4, .reveal h5, .reveal h6 { font-family:Arial, Microsoft JhengHei;} .reveal .progress { height: 14px !important; } .progress span { background: url() repeat-x !important; } .progress span:after, .progress span.nyancat { content: ""; background: url() ; width: 34px !important; height: 21px !important; border: none !important; float: right; margin-top: -7px; margin-right: -10px; } .my-icon-css { width: 100%; height: 200px; background-position:center; background-repeat: no-repeat; background-image:url('https://mir-s3-cdn-cf.behance.net/project_modules/disp/fe36cc42774743.57ee5f329fae6.gif'); } </style> <!-- .slide: data-transition="slide" --> # <h2>ch11. Testing</h2> <div class="my-icon-css"></div> @Kais(VagrantPi) ###### tags: `Rust`, `簡報`, `Rust 讀書會` --- <!-- .slide: data-transition="convex" --> ## Agenda - Writing tests - Runung tests - Test Organization --- <!-- .slide: data-transition="convex" --> ## Writing tests ---- ### assert_eq!、assert_ne! <!-- .slide: data-transition="convex" --> ```rust #[cfg(test)] mod tests { #[test] fn it_works() { assert_eq!(2 + 2, 4); } } ``` ---- <!-- .slide: data-transition="convex" --> ``` $ cargo test Compiling adder v0.1.0 (file:///projects/adder) Finished dev [unoptimized + debuginfo] target(s) in 0.22 secs Running target/debug/deps/adder-ce99bcc2479f4607 running 1 test test tests::it_works ... ok test result: ok. 1 passed; 0 failed; 0 ignored; 0 measured; 0 filtered out Doc-tests adder running 0 tests test result: ok. 0 passed; 0 failed; 0 ignored; 0 measured; 0 filtered out ``` ---- <!-- .slide: data-transition="convex" --> ``` test result: ok. 1 passed; 0 failed; 0 ignored; 0 measured; 0 filtered out ``` - passed: 通過測試 - failed: 未通過 - ignored: 忽略 - measured: 性能測試(benchmark tests),目前只有 Rust 開發版(nightly Rust)才有 - filtered out: 過濾、忽略的 ---- <!-- .slide: data-transition="convex" --> ### assert! ```rust #[derive(Debug)] pub struct Rectangle { length: u32, width: u32, } impl Rectangle { pub fn can_hold(&self, other: &Rectangle) -> bool { self.length > other.length && self.width > other.width } } #[cfg(test)] mod tests { // tests mod 如需測試外部 mod 就需要引入,這裡選擇全局導入 // 以便使用全部外部 mod 定義的內容 use super::*; #[test] fn larger_can_hold_smaller() { let larger = Rectangle { length: 8, width: 7 }; let smaller = Rectangle { length: 5, width: 1 }; assert!(larger.can_hold(&smaller)); } #[test] fn smaller_cannot_hold_larger() { let larger = Rectangle { length: 8, width: 7 }; let smaller = Rectangle { length: 5, width: 1 }; assert!(!smaller.can_hold(&larger)); } } ``` ``` running 2 tests test tests::smaller_cannot_hold_larger ... ok test tests::larger_can_hold_smaller ... ok test result: ok. 2 passed; 0 failed; 0 ignored; 0 measured; 0 filtered out ``` ---- <!-- .slide: data-transition="convex" --> ### 自訂錯誤訊息 ```diff pub fn greeting(name: &str) -> String { - format!("Hello {}!", name) + String::from("Hello!") } #[cfg(test)] mod tests { use super::*; #[test] fn greeting_contains_name() { let result = greeting("Carol"); assert!(result.contains("Carol")); } } ``` ---- <!-- .slide: data-transition="convex" --> ``` running 1 test test tests::greeting_contains_name ... FAILED failures: ---- tests::greeting_contains_name stdout ---- thread 'tests::greeting_contains_name' panicked at 'assertion failed: result.contains("Carol")', src/lib.rs:12:8 note: Run with `RUST_BACKTRACE=1` for a backtrace. failures: tests::greeting_contains_name ``` ---- <!-- .slide: data-transition="convex" --> ```rust #[test] fn greeting_contains_name() { let result = greeting("Carol"); assert!( result.contains("Carol"), "Greeting did not contain name, value was `{}`", result ); } ``` ---- <!-- .slide: data-transition="convex" --> ``` ---- tests::greeting_contains_name stdout ---- thread 'tests::greeting_contains_name' panicked at 'Greeting did not contain name, value was `Hello!`', src/lib.rs:12:8 note: Run with `RUST_BACKTRACE=1` for a backtrace. ``` ---- <!-- .slide: data-transition="convex" --> ### should_panic ```rust pub struct Guess { value: u32, } impl Guess { pub fn new(value: u32) -> Guess { if value < 1 || value > 100 { panic!("Guess value must be between 1 and 100, got {}.", value); } Guess { value } } } #[cfg(test)] mod tests { use super::*; #[test] #[should_panic] fn greater_than_100() { Guess::new(200); } } ``` ``` running 1 test test tests::greater_than_100 ... ok test result: ok. 1 passed; 0 failed; 0 ignored; 0 measured; 0 filtered out ``` ---- <!-- .slide: data-transition="convex" --> ### should_panic(expected="...") ```rust pub struct Guess { value: u32, } impl Guess { pub fn new(value: u32) -> Guess { if value < 1 { panic!("Guess value must be greater than or equal to 1, got {}.", value); } else if value > 100 { panic!("Guess value must be less than or equal to 100, got {}.", value); } Guess { value } } } #[cfg(test)] mod tests { use super::*; #[test] #[should_panic(expected = "Guess value must be less than or equal to 100")] fn greater_than_100() { Guess::new(200); } #[test] #[should_panic(expected = "Guess value must be greater than or equal to 1")] fn less_than_1() { Guess::new(0); } } ``` --- <!-- .slide: data-transition="convex" --> ## Runung tests ---- <!-- .slide: data-transition="convex" --> ## 並行執行 ``` $ cargo test -- --test-threads=1 ``` ---- <!-- .slide: data-transition="convex" --> ## 顯示函示輸出 ```rust 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); } } ``` ---- <!-- .slide: data-transition="convex" --> ``` 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:8 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 ``` 如果連成功的都要輸出,可以使用 ``` $ cargo test -- --nocapture ``` ---- <!-- .slide: data-transition="convex" --> ### 只跑單個測試 ```rust 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)); } } ``` ---- <!-- .slide: data-transition="convex" --> ``` $ 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 ``` ---- <!-- .slide: data-transition="convex" --> ### 只跑過濾過的多個測試 ``` $ 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 ``` ---- <!-- .slide: data-transition="convex" --> ### 忽略某些測試 ```rust #[test] fn it_works() { assert_eq!(2 + 2, 4); } #[test] #[ignore] fn expensive_test() { // code that takes an hour to run } ``` ---- <!-- .slide: data-transition="convex" --> ``` $ 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 ``` ---- <!-- .slide: data-transition="convex" --> 也可以只跑 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 ``` --- <!-- .slide: data-transition="convex" --> ## Test Organization ---- <!-- .slide: data-transition="convex" --> ## 單元測試(unit tests) ---- <!-- .slide: data-transition="convex" --> ### 測試 module 與 cfg(test) - `#[cfg(test)]` 註解用來告訴 Rust 只在執行 `cargo test` 時才測試,在 `cargo build` 不會做 - cfg 属性代表 configuration,用來告訴 Rust 編譯或執行時的設定 ---- <!-- .slide: data-transition="convex" --> ### 測試私有函數 src/lib.rs ```rust pub fn add_two(a: i32) -> i32 { internal_adder(a, 2) } fn internal_adder(a: i32, b: i32) -> i32 { a + b } #[cfg(test)] mod tests { use super::*; #[test] fn internal() { assert_eq!(4, internal_adder(2, 2)); } } ``` ---- <!-- .slide: data-transition="convex" --> ## 整合測試(integration tests) ---- <!-- .slide: data-transition="convex" --> 需要在專案根目錄建立 `tests` 目錄 tests/integration_test.rs ```rust // adder 為整個專案的名稱 // 在單元測試中不用寫,不過整合測試中 // 每個 tests 底下的測試檔案都是獨立的 crate // 所以需要將所每一個檔案都引入 extern crate adder; // 這裡不需要 #[cfg(test)] #[test] fn it_adds_two() { assert_eq!(4, adder::add_two(2)); } ``` ---- <!-- .slide: data-transition="convex" --> ``` $ cargo test Compiling adder v0.1.0 (file:///projects/adder) Finished dev [unoptimized + debuginfo] target(s) in 0.31 secs Running target/debug/deps/adder-abcabcabc running 1 test test tests::internal ... ok test result: ok. 1 passed; 0 failed; 0 ignored; 0 measured; 0 filtered out Running target/debug/deps/integration_test-ce99bcc2479f4607 running 1 test test it_adds_two ... ok test result: ok. 1 passed; 0 failed; 0 ignored; 0 measured; 0 filtered out Doc-tests adder running 0 tests test result: ok. 0 passed; 0 failed; 0 ignored; 0 measured; 0 filtered out ``` 輸出的三個部分分別為: 單元測試、整合測試、文件測試 ---- <!-- .slide: data-transition="convex" --> ### 只執行指定的整合測試 只跑 tests/integration_test.rs 中的測試 ``` $ cargo test --test integration_test Finished dev [unoptimized + debuginfo] target(s) in 0.0 secs Running target/debug/integration_test-952a27e0126bb565 running 1 test test it_adds_two ... ok test result: ok. 1 passed; 0 failed; 0 ignored; 0 measured; 0 filtered out ``` ---- <!-- .slide: data-transition="convex" --> ### 整合測試中的子 module tests/common.rs ```rust pub fn setup() { // setup code specific to your library's tests would go here } ``` ---- <!-- .slide: data-transition="convex" --> ``` running 1 test test tests::internal ... ok test result: ok. 1 passed; 0 failed; 0 ignored; 0 measured; 0 filtered out Running target/debug/deps/common-b8b07b6f1be2db70 running 0 tests test result: ok. 0 passed; 0 failed; 0 ignored; 0 measured; 0 filtered out Running target/debug/deps/integration_test-d993c68b431d39df running 1 test test it_adds_two ... ok test result: ok. 1 passed; 0 failed; 0 ignored; 0 measured; 0 filtered out Doc-tests adder running 0 tests test result: ok. 0 passed; 0 failed; 0 ignored; 0 measured; 0 filtered out ``` ``` 測試時不希望測試到 common.rs,我們只希望在其他整合測試時用到它 tests/common.rs -> tests/common/mod.rs ``` ---- <!-- .slide: data-transition="convex" --> tests/integration_test.rs ```rust extern crate adder; mod common; #[test] fn it_adds_two() { common::setup(); assert_eq!(4, adder::add_two(2)); } ``` --- <!-- .slide: data-transition="convex" --> ## Thanks for listening ~
{"metaMigratedAt":"2023-06-14T18:56:19.520Z","metaMigratedFrom":"Content","title":"<h2>ch11. Testing</h2>","breaks":true,"contributors":"[{\"id\":\"69ade472-3ed3-499d-8a69-767243a31621\",\"add\":19157,\"del\":1009}]"}
    1228 views