-
Notifications
You must be signed in to change notification settings - Fork 3.5k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
new chapter: testing #288
new chapter: testing #288
Changes from 16 commits
0097f8c
a85af06
62bfe3a
5428b7f
aefa195
a49d644
2f4c61d
71b87e0
182fa3d
97ffe9f
d1107ec
a68f2fa
5e49128
316db3e
e3554b7
d242dc4
b45eae6
093935f
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,30 @@ | ||
# Testing | ||
|
||
> Program testing can be a very effective way to show the presence of bugs, but | ||
> it is hopelessly inadequate for showing their absence. | ||
> | ||
> Edsger W. Dijkstra, "The Humble Programmer" (1972) | ||
|
||
Rust is a programming language that cares a lot about correctness, but | ||
correctness is a complex topic and isn't easy to prove. Rust places a lot of | ||
weight on its type system to help ensure that our programs do what we intend, | ||
but it cannot help with everything. As such, Rust also includes support for | ||
writing software tests in the language itself. | ||
|
||
For example, we can write a function called `add_two` with a signature that | ||
accepts an integer as an argument and returns an integer as a result. We can | ||
implement and compile that function, and Rust can do all the type checking and | ||
borrow checking that we've seen it's capable of doing. What Rust *can't* check | ||
for us is that we've implemented this function to return the argument plus two | ||
and not the argument plus 10 or the argument minus 50! That's where tests come | ||
in: we can write tests that, for example, pass `3` to the `add_two` function | ||
and check that we get `5` back. We can run the tests whenever we make changes | ||
to our code to make sure we didn't change any existing behavior from what the | ||
tests specify it should be. | ||
|
||
Testing is a skill, and we cannot hope to cover everything about how to write | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. The comma after skill on this line is not needed. If it was a list then having the oxford comma would be good but this is just a conjunction of two sentences. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Conjunctions of two sentences get commas, though. https://owl.english.purdue.edu/owl/resource/607/02 |
||
good tests in one chapter of a book. What we can discuss, however, are the | ||
mechanics of Rust's testing facilities. We'll talk about the annotations and | ||
macros available to you when writing your tests, the default behavior and | ||
options provided for running your tests, and how to organize tests into unit | ||
tests and integration tests. |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,279 @@ | ||
## Writing Tests | ||
|
||
Tests are Rust functions that use particular features and are written in such a | ||
way as to verify that non-test code is functioning in the expected manner. | ||
Everything we've discussed about Rust code applies to Rust tests as well! Let's | ||
look at the features Rust provides specifically for writing tests: the `test` | ||
attribute, a few macros, and the `should_panic` attribute. | ||
|
||
### The `test` attribute | ||
|
||
At its simplest, a test in Rust is a function that's annotated with the `test` | ||
attribute. Let's make a new library project with Cargo called `adder`: | ||
|
||
```text | ||
$ cargo new adder | ||
Created library `adder` project | ||
$ cd adder | ||
``` | ||
|
||
Cargo will automatically generate a simple test when you make a new library | ||
project. Here's the contents of `src/lib.rs`: | ||
|
||
Filename: src/lib.rs | ||
|
||
```rust | ||
#[cfg(test)] | ||
mod tests { | ||
#[test] | ||
fn it_works() { | ||
} | ||
} | ||
``` | ||
|
||
For now, let's ignore the `tests` module and the `#[cfg(test)]` annotation in | ||
order to focus on just the function. Note the `#[test]` before it: this | ||
attribute indicates this is a test function. The function currently has no | ||
body; that's good enough to pass! We can run the tests with `cargo test`: | ||
|
||
```text | ||
$ cargo test | ||
Compiling adder v0.1.0 (file:///projects/adder) | ||
Finished debug [unoptimized + debuginfo] target(s) in 0.22 secs | ||
Running target/debug/deps/adder-ce99bcc2479f4607 | ||
|
||
running 1 test | ||
test it_works ... ok | ||
|
||
test result: ok. 1 passed; 0 failed; 0 ignored; 0 measured | ||
|
||
Doc-tests adder | ||
|
||
running 0 tests | ||
|
||
test result: ok. 0 passed; 0 failed; 0 ignored; 0 measured | ||
``` | ||
|
||
Cargo compiled and ran our tests. There are two sets of output here; we're | ||
going to focus on the first set in this chapter. The second set of output is | ||
for documentation tests, which we'll talk about in Chapter 14. For now, note | ||
this line: | ||
|
||
```text | ||
test it_works ... ok | ||
``` | ||
|
||
The `it_works` text comes from the name of our function. | ||
|
||
We also get a summary line that tells us the aggregate results of all the | ||
tests that we have: | ||
|
||
```text | ||
test result: ok. 1 passed; 0 failed; 0 ignored; 0 measured | ||
``` | ||
|
||
### The `assert!` macro | ||
|
||
The empty test function passes because any test which doesn't `panic!` passes, | ||
and any test that does `panic!` fails. Let's make the test fail by using the | ||
`assert!` macro: | ||
|
||
Filename: src/lib.rs | ||
|
||
```rust | ||
#[test] | ||
fn it_works() { | ||
assert!(false); | ||
} | ||
``` | ||
|
||
The `assert!` macro is provided by the standard library, and it takes one | ||
argument. If the argument is `true`, nothing happens. If the argument is | ||
`false`, the macro will `panic!`. Let's run our tests again: | ||
|
||
```text | ||
$ cargo test | ||
Compiling adder v0.1.0 (file:///projects/adder) | ||
Finished debug [unoptimized + debuginfo] target(s) in 0.22 secs | ||
Running target/debug/deps/adder-ce99bcc2479f4607 | ||
|
||
running 1 test | ||
test it_works ... FAILED | ||
|
||
failures: | ||
|
||
---- it_works stdout ---- | ||
thread 'it_works' panicked at 'assertion failed: false', src/lib.rs:5 | ||
note: Run with `RUST_BACKTRACE=1` for a backtrace. | ||
|
||
|
||
failures: | ||
it_works | ||
|
||
test result: FAILED. 0 passed; 1 failed; 0 ignored; 0 measured | ||
|
||
error: test failed | ||
``` | ||
|
||
Rust indicates that our test failed: | ||
|
||
```text | ||
test it_works ... FAILED | ||
``` | ||
|
||
And shows that the test failed because the `assert!` macro in `src/lib.rs` on | ||
line 5 got a `false` value: | ||
|
||
```text | ||
thread 'it_works' panicked at 'assertion failed: false', src/lib.rs:5 | ||
``` | ||
|
||
The test failure is also reflected in the summary line: | ||
|
||
```text | ||
test result: FAILED. 0 passed; 1 failed; 0 ignored; 0 measured | ||
``` | ||
|
||
### Testing equality with the `assert_eq!` and `assert_ne!` macros | ||
|
||
A common way to test functionality is to compare the result of the code under | ||
test to the value you expect it to be, and check that they're equal. You can do | ||
this using the `assert!` macro by passing it an expression using the `==` | ||
macro. This is so common, though, that the standard library provides a pair of | ||
macros to do this for convenience: `assert_eq!` and `assert_ne!`. These macros | ||
compare two arguments for equality or inequality, respectively. The other | ||
advantage of using these macros is they will print out what the two values | ||
actually are if the assertion fails so that it's easier to see *why* the test | ||
failed, whereas the `assert!` macro would just print out that it got a `false` | ||
value for the `==` expression. | ||
|
||
Here's an example test that uses each of these macros and will pass: | ||
|
||
Filename: src/lib.rs | ||
|
||
```rust | ||
#[test] | ||
fn it_works() { | ||
assert_eq!("Hello", "Hello"); | ||
|
||
assert_ne!("Hello", "world"); | ||
} | ||
``` | ||
|
||
You can also specify an optional third argument to each of these macros, which | ||
is a custom message that you'd like to be added to the failure message. The | ||
macros expand to logic similar to this: | ||
|
||
```rust,ignore | ||
// assert_eq! - panic if the values aren't equal | ||
if left_val != right_val { | ||
panic!( | ||
"assertion failed: `(left == right)` (left: `{:?}`, right: `{:?}`): {}" | ||
left_val, | ||
right_val, | ||
optional_custom_message | ||
) | ||
} | ||
|
||
// assert_ne! - panic if the values are equal | ||
if left_val == right_val { | ||
panic!( | ||
"assertion failed: `(left != right)` (left: `{:?}`, right: `{:?}`): {}" | ||
left_val, | ||
right_val, | ||
optional_custom_message | ||
) | ||
} | ||
``` | ||
|
||
Let's take a look at a test that will fail becasue `hello` is not equal to | ||
`world`. We've also added a custom error message, `greeting operation failed`: | ||
|
||
Filename: src/lib.rs | ||
|
||
```rust | ||
#[test] | ||
fn a_simple_case() { | ||
let result = "hello"; // this value would come from running your code | ||
assert_eq!(result, "world", "greeting operation failed"); | ||
} | ||
``` | ||
|
||
Running this indeed fails, and the output we get explains why the test failed | ||
and includes the custom error message we specified: | ||
|
||
```text | ||
---- a_simple_case stdout ---- | ||
thread 'a_simple_case' panicked at 'assertion failed: `(left == right)` (left: `"hello"`, right: `"world"`): greeting operation failed', src/main.rs:4 | ||
``` | ||
|
||
The two arguments to `assert_eq!` are named "left" and "right" rather than | ||
"expected" and "actual"; the order of the value that comes from your code and | ||
the value hardcoded into your test isn't important. | ||
|
||
Since these macros use the operators `==` and `!=` and print the values using | ||
debug formatting, the values being compared must implement the `PartialEq` and | ||
`Debug` traits. Types provided by Rust implement these traits, but for structs | ||
and enums that you define, you'll need to add `PartialEq` in order to be able | ||
to assert that values of those types are equal or not equal and `Debug` in | ||
order to be able to print out the values in the case that the assertion fails. | ||
Because both of these traits are derivable traits that we mentioned in Chapter | ||
5, usually this is as straightforward as adding the `#[derive(PartialEq, | ||
Debug)]` annotation to your struct or enum definition. See Appendix C for more | ||
details about these and other derivable traits. | ||
|
||
## Test for failure with `should_panic` | ||
|
||
We can invert our test's failure with another attribute: `should_panic`. This | ||
is useful when we want to test that calling a particular function will cause an | ||
error. For example, let's test something that we know will panic from Chapter | ||
8: attempting to create a slice using range syntax with byte indices that | ||
aren't on character boundaries. Add the `#[should_panic]` attribute before the | ||
function like the `#[test]` attribute, as shown in Listing 11-1: | ||
|
||
Filename: src/lib.rs | ||
|
||
```rust | ||
#[test] | ||
#[should_panic] | ||
fn slice_not_on_char_boundaries() { | ||
let s = "Здравствуйте"; | ||
&s[0..1]; | ||
} | ||
``` | ||
|
||
<caption> | ||
Listing 11-1: A test expecting a `panic!` | ||
</caption> | ||
|
||
This test will succeed, since the code panics and we said that it should. If | ||
this code happened to run and did not cause a `panic!`, this test would fail. | ||
|
||
`should_panic` tests can be fragile, as it's hard to guarantee that the test | ||
didn't fail for a different reason than the one you were expecting. To help | ||
with this, an optional `expected` parameter can be added to the `should_panic` | ||
attribute. The test harness will make sure that the failure message contains | ||
the provided text. A safer version of Listing 11-1 would be the following, in | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. "safe" is probably a bad word here, given rust's focus on a different kind of safety. What about "more robust"? |
||
Listing 11-2: | ||
|
||
Filename: src/lib.rs | ||
|
||
```rust | ||
#[test] | ||
#[should_panic(expected = "do not lie on character boundary")] | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. (we should leave this for now, but it'll need fixed at some point) |
||
fn slice_not_on_char_boundaries() { | ||
let s = "Здравствуйте"; | ||
&s[0..1]; | ||
} | ||
``` | ||
|
||
<!-- I will add ghosting in libreoffice /Carol --> | ||
|
||
<caption> | ||
Listing 11-2: A test expecting a `panic!` with a particular message | ||
</caption> | ||
|
||
Try on your own to see what happens when a `should_panic` test panics but | ||
doesn't match the expected message: cause a `panic!` that happens for a | ||
different reason in this test, or change the expected panic message to | ||
something that doesn't match the character boundary panic message. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
You might just want to write this as 'where tests come in. We can write tests that'
Having the : made the flow a bit weird. That or 'come in: we can write, for example' using tests as a word twice so close to each other trips the flow up.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
¯\_(ツ)_/¯ none of these versions bother me, so might as well change it :)