2015-02-09 22:24:49 -05:00
|
|
|
|
% Documentation
|
2015-01-21 14:59:25 -05:00
|
|
|
|
|
2015-02-19 15:35:26 -05:00
|
|
|
|
Documentation is an important part of any software project, and it's
|
|
|
|
|
first-class in Rust. Let's talk about the tooling Rust gives you to
|
|
|
|
|
document your project.
|
2015-01-21 14:59:25 -05:00
|
|
|
|
|
2015-02-19 15:35:26 -05:00
|
|
|
|
## About `rustdoc`
|
2015-01-21 14:59:25 -05:00
|
|
|
|
|
2015-02-19 15:35:26 -05:00
|
|
|
|
The Rust distribution includes a tool, `rustdoc`, that generates documentation.
|
|
|
|
|
`rustdoc` is also used by Cargo through `cargo doc`.
|
2015-01-21 14:59:25 -05:00
|
|
|
|
|
2015-02-19 15:35:26 -05:00
|
|
|
|
Documentation can be generated in two ways: from source code, and from
|
|
|
|
|
standalone Markdown files.
|
2015-01-21 14:59:25 -05:00
|
|
|
|
|
2015-02-19 15:35:26 -05:00
|
|
|
|
## Documenting source code
|
2015-01-21 14:59:25 -05:00
|
|
|
|
|
2015-02-19 15:35:26 -05:00
|
|
|
|
The primary way of documenting a Rust project is through annotating the source
|
|
|
|
|
code. You can use documentation comments for this purpose:
|
2015-01-21 14:59:25 -05:00
|
|
|
|
|
2015-02-19 15:35:26 -05:00
|
|
|
|
```rust,ignore
|
|
|
|
|
/// Constructs a new `Rc<T>`.
|
|
|
|
|
///
|
|
|
|
|
/// # Examples
|
|
|
|
|
///
|
|
|
|
|
/// ```
|
|
|
|
|
/// use std::rc::Rc;
|
|
|
|
|
///
|
|
|
|
|
/// let five = Rc::new(5);
|
|
|
|
|
/// ```
|
|
|
|
|
pub fn new(value: T) -> Rc<T> {
|
|
|
|
|
// implementation goes here
|
|
|
|
|
}
|
|
|
|
|
```
|
2015-01-21 14:59:25 -05:00
|
|
|
|
|
2015-02-19 15:35:26 -05:00
|
|
|
|
This code generates documentation that looks [like this][rc-new]. I've left the
|
2015-06-22 19:26:51 +02:00
|
|
|
|
implementation out, with a regular comment in its place.
|
|
|
|
|
|
|
|
|
|
The first thing to notice about this annotation is that it uses
|
|
|
|
|
`///` instead of `//`. The triple slash
|
2015-02-19 15:35:26 -05:00
|
|
|
|
indicates a documentation comment.
|
2015-01-21 14:59:25 -05:00
|
|
|
|
|
2015-02-19 15:35:26 -05:00
|
|
|
|
Documentation comments are written in Markdown.
|
|
|
|
|
|
|
|
|
|
Rust keeps track of these comments, and uses them when generating
|
|
|
|
|
documentation. This is important when documenting things like enums:
|
2015-01-21 14:59:25 -05:00
|
|
|
|
|
2015-05-18 20:56:00 +02:00
|
|
|
|
```rust
|
2015-09-30 12:33:38 -04:00
|
|
|
|
/// The `Option` type. See [the module level documentation](index.html) for more.
|
2015-02-19 15:35:26 -05:00
|
|
|
|
enum Option<T> {
|
|
|
|
|
/// No value
|
|
|
|
|
None,
|
|
|
|
|
/// Some value `T`
|
|
|
|
|
Some(T),
|
|
|
|
|
}
|
2015-01-21 14:59:25 -05:00
|
|
|
|
```
|
|
|
|
|
|
2015-02-19 15:35:26 -05:00
|
|
|
|
The above works, but this does not:
|
2015-01-21 14:59:25 -05:00
|
|
|
|
|
2015-02-19 15:35:26 -05:00
|
|
|
|
```rust,ignore
|
2015-09-30 12:33:38 -04:00
|
|
|
|
/// The `Option` type. See [the module level documentation](index.html) for more.
|
2015-02-19 15:35:26 -05:00
|
|
|
|
enum Option<T> {
|
|
|
|
|
None, /// No value
|
|
|
|
|
Some(T), /// Some value `T`
|
|
|
|
|
}
|
|
|
|
|
```
|
2015-01-21 14:59:25 -05:00
|
|
|
|
|
2015-02-19 15:35:26 -05:00
|
|
|
|
You'll get an error:
|
2015-01-21 14:59:25 -05:00
|
|
|
|
|
2015-02-19 15:35:26 -05:00
|
|
|
|
```text
|
|
|
|
|
hello.rs:4:1: 4:2 error: expected ident, found `}`
|
|
|
|
|
hello.rs:4 }
|
|
|
|
|
^
|
|
|
|
|
```
|
2015-01-21 14:59:25 -05:00
|
|
|
|
|
2015-02-19 15:35:26 -05:00
|
|
|
|
This [unfortunate error](https://github.com/rust-lang/rust/issues/22547) is
|
2015-10-19 11:16:31 -04:00
|
|
|
|
correct: documentation comments apply to the thing after them, and there's
|
2015-08-21 17:10:46 +01:00
|
|
|
|
nothing after that last comment.
|
2015-01-21 14:59:25 -05:00
|
|
|
|
|
2015-08-09 14:15:05 -07:00
|
|
|
|
[rc-new]: https://doc.rust-lang.org/nightly/std/rc/struct.Rc.html#method.new
|
2015-01-21 14:59:25 -05:00
|
|
|
|
|
2015-02-19 15:35:26 -05:00
|
|
|
|
### Writing documentation comments
|
2015-01-21 14:59:25 -05:00
|
|
|
|
|
2015-02-19 15:35:26 -05:00
|
|
|
|
Anyway, let's cover each part of this comment in detail:
|
2015-01-21 14:59:25 -05:00
|
|
|
|
|
2015-05-18 20:56:00 +02:00
|
|
|
|
```rust
|
2015-02-19 15:35:26 -05:00
|
|
|
|
/// Constructs a new `Rc<T>`.
|
|
|
|
|
# fn foo() {}
|
|
|
|
|
```
|
2015-01-21 14:59:25 -05:00
|
|
|
|
|
2015-02-19 15:35:26 -05:00
|
|
|
|
The first line of a documentation comment should be a short summary of its
|
|
|
|
|
functionality. One sentence. Just the basics. High level.
|
2015-01-21 14:59:25 -05:00
|
|
|
|
|
2015-05-18 20:56:00 +02:00
|
|
|
|
```rust
|
2015-02-19 15:35:26 -05:00
|
|
|
|
///
|
|
|
|
|
/// Other details about constructing `Rc<T>`s, maybe describing complicated
|
|
|
|
|
/// semantics, maybe additional options, all kinds of stuff.
|
|
|
|
|
///
|
|
|
|
|
# fn foo() {}
|
|
|
|
|
```
|
2015-01-21 14:59:25 -05:00
|
|
|
|
|
2015-02-19 15:35:26 -05:00
|
|
|
|
Our original example had just a summary line, but if we had more things to say,
|
|
|
|
|
we could have added more explanation in a new paragraph.
|
2015-01-21 14:59:25 -05:00
|
|
|
|
|
2015-02-19 15:35:26 -05:00
|
|
|
|
#### Special sections
|
2015-01-21 14:59:25 -05:00
|
|
|
|
|
2015-02-19 15:35:26 -05:00
|
|
|
|
Next, are special sections. These are indicated with a header, `#`. There
|
2015-06-22 20:22:31 +02:00
|
|
|
|
are four kinds of headers that are commonly used. They aren't special syntax,
|
2015-02-19 15:35:26 -05:00
|
|
|
|
just convention, for now.
|
2015-01-21 14:59:25 -05:00
|
|
|
|
|
2015-05-18 20:56:00 +02:00
|
|
|
|
```rust
|
2015-02-19 15:35:26 -05:00
|
|
|
|
/// # Panics
|
|
|
|
|
# fn foo() {}
|
|
|
|
|
```
|
2015-01-21 14:59:25 -05:00
|
|
|
|
|
2015-02-19 15:35:26 -05:00
|
|
|
|
Unrecoverable misuses of a function (i.e. programming errors) in Rust are
|
|
|
|
|
usually indicated by panics, which kill the whole current thread at the very
|
|
|
|
|
least. If your function has a non-trivial contract like this, that is
|
|
|
|
|
detected/enforced by panics, documenting it is very important.
|
2015-01-21 14:59:25 -05:00
|
|
|
|
|
2015-05-18 20:56:00 +02:00
|
|
|
|
```rust
|
2015-02-19 15:35:26 -05:00
|
|
|
|
/// # Failures
|
|
|
|
|
# fn foo() {}
|
2015-01-21 14:59:25 -05:00
|
|
|
|
```
|
|
|
|
|
|
2015-02-19 15:35:26 -05:00
|
|
|
|
If your function or method returns a `Result<T, E>`, then describing the
|
|
|
|
|
conditions under which it returns `Err(E)` is a nice thing to do. This is
|
|
|
|
|
slightly less important than `Panics`, because failure is encoded into the type
|
|
|
|
|
system, but it's still a good thing to do.
|
2015-01-21 14:59:25 -05:00
|
|
|
|
|
2015-05-18 20:56:00 +02:00
|
|
|
|
```rust
|
2015-02-19 15:35:26 -05:00
|
|
|
|
/// # Safety
|
|
|
|
|
# fn foo() {}
|
2015-01-21 14:59:25 -05:00
|
|
|
|
```
|
|
|
|
|
|
2015-02-19 15:35:26 -05:00
|
|
|
|
If your function is `unsafe`, you should explain which invariants the caller is
|
|
|
|
|
responsible for upholding.
|
2015-01-21 14:59:25 -05:00
|
|
|
|
|
2015-05-18 20:56:00 +02:00
|
|
|
|
```rust
|
2015-02-19 15:35:26 -05:00
|
|
|
|
/// # Examples
|
|
|
|
|
///
|
|
|
|
|
/// ```
|
|
|
|
|
/// use std::rc::Rc;
|
|
|
|
|
///
|
|
|
|
|
/// let five = Rc::new(5);
|
|
|
|
|
/// ```
|
|
|
|
|
# fn foo() {}
|
|
|
|
|
```
|
2015-01-21 14:59:25 -05:00
|
|
|
|
|
2015-06-22 20:22:31 +02:00
|
|
|
|
Fourth, `Examples`. Include one or more examples of using your function or
|
2015-02-19 15:35:26 -05:00
|
|
|
|
method, and your users will love you for it. These examples go inside of
|
|
|
|
|
code block annotations, which we'll talk about in a moment, and can have
|
|
|
|
|
more than one section:
|
2015-01-21 14:59:25 -05:00
|
|
|
|
|
2015-05-18 20:56:00 +02:00
|
|
|
|
```rust
|
2015-02-19 15:35:26 -05:00
|
|
|
|
/// # Examples
|
|
|
|
|
///
|
|
|
|
|
/// Simple `&str` patterns:
|
|
|
|
|
///
|
|
|
|
|
/// ```
|
|
|
|
|
/// let v: Vec<&str> = "Mary had a little lamb".split(' ').collect();
|
|
|
|
|
/// assert_eq!(v, vec!["Mary", "had", "a", "little", "lamb"]);
|
|
|
|
|
/// ```
|
|
|
|
|
///
|
|
|
|
|
/// More complex patterns with a lambda:
|
|
|
|
|
///
|
|
|
|
|
/// ```
|
|
|
|
|
/// let v: Vec<&str> = "abc1def2ghi".split(|c: char| c.is_numeric()).collect();
|
|
|
|
|
/// assert_eq!(v, vec!["abc", "def", "ghi"]);
|
|
|
|
|
/// ```
|
|
|
|
|
# fn foo() {}
|
|
|
|
|
```
|
2015-01-21 14:59:25 -05:00
|
|
|
|
|
2015-02-19 15:35:26 -05:00
|
|
|
|
Let's discuss the details of these code blocks.
|
2015-01-21 14:59:25 -05:00
|
|
|
|
|
2015-02-19 15:35:26 -05:00
|
|
|
|
#### Code block annotations
|
2015-01-21 14:59:25 -05:00
|
|
|
|
|
2015-02-19 15:35:26 -05:00
|
|
|
|
To write some Rust code in a comment, use the triple graves:
|
2015-01-21 14:59:25 -05:00
|
|
|
|
|
2015-05-18 20:56:00 +02:00
|
|
|
|
```rust
|
2015-02-19 15:35:26 -05:00
|
|
|
|
/// ```
|
|
|
|
|
/// println!("Hello, world");
|
|
|
|
|
/// ```
|
|
|
|
|
# fn foo() {}
|
|
|
|
|
```
|
2015-01-21 14:59:25 -05:00
|
|
|
|
|
2015-02-19 15:35:26 -05:00
|
|
|
|
If you want something that's not Rust code, you can add an annotation:
|
2015-01-21 14:59:25 -05:00
|
|
|
|
|
2015-05-18 20:56:00 +02:00
|
|
|
|
```rust
|
2015-02-19 15:35:26 -05:00
|
|
|
|
/// ```c
|
|
|
|
|
/// printf("Hello, world\n");
|
|
|
|
|
/// ```
|
|
|
|
|
# fn foo() {}
|
2015-01-21 14:59:25 -05:00
|
|
|
|
```
|
|
|
|
|
|
2015-02-19 15:35:26 -05:00
|
|
|
|
This will highlight according to whatever language you're showing off.
|
|
|
|
|
If you're just showing plain text, choose `text`.
|
|
|
|
|
|
|
|
|
|
It's important to choose the correct annotation here, because `rustdoc` uses it
|
2015-08-20 18:41:24 -04:00
|
|
|
|
in an interesting way: It can be used to actually test your examples in a
|
|
|
|
|
library crate, so that they don't get out of date. If you have some C code but
|
|
|
|
|
`rustdoc` thinks it's Rust because you left off the annotation, `rustdoc` will
|
|
|
|
|
complain when trying to generate the documentation.
|
2015-01-21 14:59:25 -05:00
|
|
|
|
|
2015-02-19 15:35:26 -05:00
|
|
|
|
## Documentation as tests
|
|
|
|
|
|
|
|
|
|
Let's discuss our sample example documentation:
|
|
|
|
|
|
2015-05-18 20:56:00 +02:00
|
|
|
|
```rust
|
2015-02-19 15:35:26 -05:00
|
|
|
|
/// ```
|
|
|
|
|
/// println!("Hello, world");
|
|
|
|
|
/// ```
|
|
|
|
|
# fn foo() {}
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
You'll notice that you don't need a `fn main()` or anything here. `rustdoc` will
|
2015-10-13 15:59:59 +09:00
|
|
|
|
automatically add a `main()` wrapper around your code, and in the right place.
|
2015-02-19 15:35:26 -05:00
|
|
|
|
For example:
|
2015-01-21 14:59:25 -05:00
|
|
|
|
|
2015-05-18 20:56:00 +02:00
|
|
|
|
```rust
|
2015-01-21 14:59:25 -05:00
|
|
|
|
/// ```
|
|
|
|
|
/// use std::rc::Rc;
|
|
|
|
|
///
|
|
|
|
|
/// let five = Rc::new(5);
|
|
|
|
|
/// ```
|
|
|
|
|
# fn foo() {}
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
This will end up testing:
|
|
|
|
|
|
2015-05-18 20:56:00 +02:00
|
|
|
|
```rust
|
2015-01-21 14:59:25 -05:00
|
|
|
|
fn main() {
|
|
|
|
|
use std::rc::Rc;
|
|
|
|
|
let five = Rc::new(5);
|
|
|
|
|
}
|
|
|
|
|
```
|
|
|
|
|
|
2015-10-13 15:59:59 +09:00
|
|
|
|
Here's the full algorithm rustdoc uses to preprocess examples:
|
2015-03-12 13:01:06 -07:00
|
|
|
|
|
|
|
|
|
1. Any leading `#![foo]` attributes are left intact as crate attributes.
|
|
|
|
|
2. Some common `allow` attributes are inserted, including
|
|
|
|
|
`unused_variables`, `unused_assignments`, `unused_mut`,
|
|
|
|
|
`unused_attributes`, and `dead_code`. Small examples often trigger
|
|
|
|
|
these lints.
|
|
|
|
|
3. If the example does not contain `extern crate`, then `extern crate
|
|
|
|
|
<mycrate>;` is inserted.
|
|
|
|
|
2. Finally, if the example does not contain `fn main`, the remainder of the
|
|
|
|
|
text is wrapped in `fn main() { your_code }`
|
2015-02-19 15:35:26 -05:00
|
|
|
|
|
|
|
|
|
Sometimes, this isn't enough, though. For example, all of these code samples
|
|
|
|
|
with `///` we've been talking about? The raw text:
|
|
|
|
|
|
|
|
|
|
```text
|
|
|
|
|
/// Some documentation.
|
|
|
|
|
# fn foo() {}
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
looks different than the output:
|
|
|
|
|
|
2015-05-18 20:56:00 +02:00
|
|
|
|
```rust
|
2015-02-19 15:35:26 -05:00
|
|
|
|
/// Some documentation.
|
|
|
|
|
# fn foo() {}
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
Yes, that's right: you can add lines that start with `# `, and they will
|
|
|
|
|
be hidden from the output, but will be used when compiling your code. You
|
|
|
|
|
can use this to your advantage. In this case, documentation comments need
|
|
|
|
|
to apply to some kind of function, so if I want to show you just a
|
|
|
|
|
documentation comment, I need to add a little function definition below
|
|
|
|
|
it. At the same time, it's just there to satisfy the compiler, so hiding
|
|
|
|
|
it makes the example more clear. You can use this technique to explain
|
|
|
|
|
longer examples in detail, while still preserving the testability of your
|
2015-10-25 15:58:50 -04:00
|
|
|
|
documentation.
|
2015-02-19 15:35:26 -05:00
|
|
|
|
|
2015-10-25 15:58:50 -04:00
|
|
|
|
For example, imagine that we wanted to document this code:
|
2015-02-19 15:35:26 -05:00
|
|
|
|
|
2015-05-18 20:56:00 +02:00
|
|
|
|
```rust
|
2015-02-19 15:35:26 -05:00
|
|
|
|
let x = 5;
|
|
|
|
|
let y = 6;
|
|
|
|
|
println!("{}", x + y);
|
|
|
|
|
```
|
|
|
|
|
|
2015-10-25 15:58:50 -04:00
|
|
|
|
We might want the documentation to end up looking like this:
|
2015-10-02 18:54:07 -04:00
|
|
|
|
|
2015-02-19 15:35:26 -05:00
|
|
|
|
> First, we set `x` to five:
|
2015-03-14 16:09:26 -07:00
|
|
|
|
>
|
2015-10-25 15:58:50 -04:00
|
|
|
|
> ```rust
|
2015-02-19 15:35:26 -05:00
|
|
|
|
> let x = 5;
|
|
|
|
|
> # let y = 6;
|
|
|
|
|
> # println!("{}", x + y);
|
|
|
|
|
> ```
|
2015-03-14 16:09:26 -07:00
|
|
|
|
>
|
2015-02-19 15:35:26 -05:00
|
|
|
|
> Next, we set `y` to six:
|
2015-03-14 16:09:26 -07:00
|
|
|
|
>
|
2015-10-25 15:58:50 -04:00
|
|
|
|
> ```rust
|
2015-02-19 15:35:26 -05:00
|
|
|
|
> # let x = 5;
|
|
|
|
|
> let y = 6;
|
|
|
|
|
> # println!("{}", x + y);
|
|
|
|
|
> ```
|
2015-03-14 16:09:26 -07:00
|
|
|
|
>
|
2015-02-19 15:35:26 -05:00
|
|
|
|
> Finally, we print the sum of `x` and `y`:
|
2015-03-14 16:09:26 -07:00
|
|
|
|
>
|
2015-10-25 15:58:50 -04:00
|
|
|
|
> ```rust
|
2015-02-19 15:35:26 -05:00
|
|
|
|
> # let x = 5;
|
|
|
|
|
> # let y = 6;
|
|
|
|
|
> println!("{}", x + y);
|
|
|
|
|
> ```
|
|
|
|
|
|
2015-10-25 15:58:50 -04:00
|
|
|
|
To keep each code block testable, we want the whole program in each block, but
|
|
|
|
|
we don't want the reader to see every line every time. Here's what we put in
|
|
|
|
|
our source code:
|
|
|
|
|
|
|
|
|
|
```text
|
|
|
|
|
First, we set `x` to five:
|
|
|
|
|
|
|
|
|
|
```text
|
|
|
|
|
let x = 5;
|
|
|
|
|
# let y = 6;
|
|
|
|
|
# println!("{}", x + y);
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
Next, we set `y` to six:
|
|
|
|
|
|
|
|
|
|
```text
|
|
|
|
|
# let x = 5;
|
|
|
|
|
let y = 6;
|
|
|
|
|
# println!("{}", x + y);
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
Finally, we print the sum of `x` and `y`:
|
|
|
|
|
|
|
|
|
|
```text
|
|
|
|
|
# let x = 5;
|
|
|
|
|
# let y = 6;
|
|
|
|
|
println!("{}", x + y);
|
|
|
|
|
```
|
|
|
|
|
```
|
2015-10-02 18:54:07 -04:00
|
|
|
|
|
2015-02-19 15:35:26 -05:00
|
|
|
|
By repeating all parts of the example, you can ensure that your example still
|
|
|
|
|
compiles, while only showing the parts that are relevant to that part of your
|
|
|
|
|
explanation.
|
|
|
|
|
|
2015-03-22 15:26:23 -04:00
|
|
|
|
### Documenting macros
|
|
|
|
|
|
|
|
|
|
Here’s an example of documenting a macro:
|
|
|
|
|
|
2015-05-18 20:56:00 +02:00
|
|
|
|
```rust
|
2015-03-22 15:26:23 -04:00
|
|
|
|
/// Panic with a given message unless an expression evaluates to true.
|
|
|
|
|
///
|
|
|
|
|
/// # Examples
|
|
|
|
|
///
|
|
|
|
|
/// ```
|
|
|
|
|
/// # #[macro_use] extern crate foo;
|
|
|
|
|
/// # fn main() {
|
|
|
|
|
/// panic_unless!(1 + 1 == 2, “Math is broken.”);
|
|
|
|
|
/// # }
|
|
|
|
|
/// ```
|
|
|
|
|
///
|
2015-03-26 13:30:33 -07:00
|
|
|
|
/// ```should_panic
|
2015-03-22 15:26:23 -04:00
|
|
|
|
/// # #[macro_use] extern crate foo;
|
|
|
|
|
/// # fn main() {
|
|
|
|
|
/// panic_unless!(true == false, “I’m broken.”);
|
|
|
|
|
/// # }
|
|
|
|
|
/// ```
|
|
|
|
|
#[macro_export]
|
|
|
|
|
macro_rules! panic_unless {
|
|
|
|
|
($condition:expr, $($rest:expr),+) => ({ if ! $condition { panic!($($rest),+); } });
|
2015-03-23 15:54:39 -07:00
|
|
|
|
}
|
|
|
|
|
# fn main() {}
|
2015-03-22 15:26:23 -04:00
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
You’ll note three things: we need to add our own `extern crate` line, so that
|
|
|
|
|
we can add the `#[macro_use]` attribute. Second, we’ll need to add our own
|
|
|
|
|
`main()` as well. Finally, a judicious use of `#` to comment out those two
|
|
|
|
|
things, so they don’t show up in the output.
|
|
|
|
|
|
2015-10-28 10:39:22 +09:00
|
|
|
|
Another case where the use of `#` is handy is when you want to ignore
|
|
|
|
|
error handling. Lets say you want the following,
|
|
|
|
|
|
2015-10-29 09:43:54 +09:00
|
|
|
|
```rust,ignore
|
2015-10-28 10:39:22 +09:00
|
|
|
|
/// use std::io;
|
|
|
|
|
/// let mut input = String::new();
|
|
|
|
|
/// try!(io::stdin().read_line(&mut input));
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
The problem is that `try!` returns a `Result<T, E>` and test functions
|
|
|
|
|
don't return anything so this will give a mismatched types error.
|
|
|
|
|
|
|
|
|
|
```rust
|
|
|
|
|
/// A doc test using try!
|
|
|
|
|
///
|
|
|
|
|
/// ```
|
|
|
|
|
/// use std::io;
|
|
|
|
|
/// # fn f() -> io::Result<()> {
|
|
|
|
|
/// let mut input = String::new();
|
|
|
|
|
/// try!(io::stdin().read_line(&mut input));
|
|
|
|
|
/// # Ok(())
|
|
|
|
|
/// # }
|
|
|
|
|
/// # f();
|
|
|
|
|
/// ```
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
You can get around this by wrapping the code in a function. This catches
|
|
|
|
|
and swallows the `Result<T, E>` when running tests on the docs. This
|
|
|
|
|
pattern appears regularly in the standard library.
|
|
|
|
|
|
2015-03-22 15:26:23 -04:00
|
|
|
|
### Running documentation tests
|
|
|
|
|
|
2015-10-13 15:59:59 +09:00
|
|
|
|
To run the tests, either:
|
2015-02-19 15:35:26 -05:00
|
|
|
|
|
|
|
|
|
```bash
|
|
|
|
|
$ rustdoc --test path/to/my/crate/root.rs
|
|
|
|
|
# or
|
|
|
|
|
$ cargo test
|
|
|
|
|
```
|
|
|
|
|
|
2015-08-20 18:41:24 -04:00
|
|
|
|
That's right, `cargo test` tests embedded documentation too. **However,
|
|
|
|
|
`cargo test` will not test binary crates, only library ones.** This is
|
2015-04-16 15:55:10 -04:00
|
|
|
|
due to the way `rustdoc` works: it links against the library to be tested,
|
|
|
|
|
but with a binary, there’s nothing to link to.
|
2015-02-19 15:35:26 -05:00
|
|
|
|
|
|
|
|
|
There are a few more annotations that are useful to help `rustdoc` do the right
|
|
|
|
|
thing when testing your code:
|
|
|
|
|
|
2015-05-18 20:56:00 +02:00
|
|
|
|
```rust
|
2015-02-19 15:35:26 -05:00
|
|
|
|
/// ```ignore
|
|
|
|
|
/// fn foo() {
|
|
|
|
|
/// ```
|
|
|
|
|
# fn foo() {}
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
The `ignore` directive tells Rust to ignore your code. This is almost never
|
|
|
|
|
what you want, as it's the most generic. Instead, consider annotating it
|
|
|
|
|
with `text` if it's not code, or using `#`s to get a working example that
|
|
|
|
|
only shows the part you care about.
|
|
|
|
|
|
2015-05-18 20:56:00 +02:00
|
|
|
|
```rust
|
2015-02-19 15:35:26 -05:00
|
|
|
|
/// ```should_panic
|
|
|
|
|
/// assert!(false);
|
|
|
|
|
/// ```
|
|
|
|
|
# fn foo() {}
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
`should_panic` tells `rustdoc` that the code should compile correctly, but
|
|
|
|
|
not actually pass as a test.
|
|
|
|
|
|
2015-05-18 20:56:00 +02:00
|
|
|
|
```rust
|
2015-02-19 15:35:26 -05:00
|
|
|
|
/// ```no_run
|
|
|
|
|
/// loop {
|
|
|
|
|
/// println!("Hello, world");
|
|
|
|
|
/// }
|
|
|
|
|
/// ```
|
|
|
|
|
# fn foo() {}
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
The `no_run` attribute will compile your code, but not run it. This is
|
|
|
|
|
important for examples such as "Here's how to start up a network service,"
|
|
|
|
|
which you would want to make sure compile, but might run in an infinite loop!
|
|
|
|
|
|
|
|
|
|
### Documenting modules
|
|
|
|
|
|
|
|
|
|
Rust has another kind of doc comment, `//!`. This comment doesn't document the next item, but the enclosing item. In other words:
|
|
|
|
|
|
2015-05-18 20:56:00 +02:00
|
|
|
|
```rust
|
2015-02-19 15:35:26 -05:00
|
|
|
|
mod foo {
|
|
|
|
|
//! This is documentation for the `foo` module.
|
|
|
|
|
//!
|
|
|
|
|
//! # Examples
|
|
|
|
|
|
|
|
|
|
// ...
|
|
|
|
|
}
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
This is where you'll see `//!` used most often: for module documentation. If
|
|
|
|
|
you have a module in `foo.rs`, you'll often open its code and see this:
|
|
|
|
|
|
2015-05-18 20:56:00 +02:00
|
|
|
|
```rust
|
2015-02-19 15:35:26 -05:00
|
|
|
|
//! A module for using `foo`s.
|
|
|
|
|
//!
|
|
|
|
|
//! The `foo` module contains a lot of useful functionality blah blah blah
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
### Documentation comment style
|
|
|
|
|
|
|
|
|
|
Check out [RFC 505][rfc505] for full conventions around the style and format of
|
|
|
|
|
documentation.
|
|
|
|
|
|
|
|
|
|
[rfc505]: https://github.com/rust-lang/rfcs/blob/master/text/0505-api-comment-conventions.md
|
|
|
|
|
|
|
|
|
|
## Other documentation
|
|
|
|
|
|
|
|
|
|
All of this behavior works in non-Rust source files too. Because comments
|
|
|
|
|
are written in Markdown, they're often `.md` files.
|
|
|
|
|
|
|
|
|
|
When you write documentation in Markdown files, you don't need to prefix
|
|
|
|
|
the documentation with comments. For example:
|
2015-01-21 14:59:25 -05:00
|
|
|
|
|
2015-05-18 20:56:00 +02:00
|
|
|
|
```rust
|
2015-02-19 15:35:26 -05:00
|
|
|
|
/// # Examples
|
|
|
|
|
///
|
|
|
|
|
/// ```
|
|
|
|
|
/// use std::rc::Rc;
|
|
|
|
|
///
|
|
|
|
|
/// let five = Rc::new(5);
|
|
|
|
|
/// ```
|
|
|
|
|
# fn foo() {}
|
|
|
|
|
```
|
2015-01-21 14:59:25 -05:00
|
|
|
|
|
2015-02-19 15:35:26 -05:00
|
|
|
|
is just
|
2015-01-21 14:59:25 -05:00
|
|
|
|
|
2015-02-19 15:35:26 -05:00
|
|
|
|
~~~markdown
|
|
|
|
|
# Examples
|
2015-01-21 14:59:25 -05:00
|
|
|
|
|
2015-02-19 15:35:26 -05:00
|
|
|
|
```
|
|
|
|
|
use std::rc::Rc;
|
|
|
|
|
|
|
|
|
|
let five = Rc::new(5);
|
|
|
|
|
```
|
2015-01-21 14:59:25 -05:00
|
|
|
|
~~~
|
|
|
|
|
|
2015-02-19 15:35:26 -05:00
|
|
|
|
when it's in a Markdown file. There is one wrinkle though: Markdown files need
|
|
|
|
|
to have a title like this:
|
2015-01-21 14:59:25 -05:00
|
|
|
|
|
2015-02-19 15:35:26 -05:00
|
|
|
|
```markdown
|
|
|
|
|
% The title
|
2015-01-21 14:59:25 -05:00
|
|
|
|
|
2015-02-19 15:35:26 -05:00
|
|
|
|
This is the example documentation.
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
This `%` line needs to be the very first line of the file.
|
|
|
|
|
|
|
|
|
|
## `doc` attributes
|
|
|
|
|
|
2015-10-13 15:59:59 +09:00
|
|
|
|
At a deeper level, documentation comments are syntactic sugar for documentation
|
|
|
|
|
attributes:
|
2015-02-19 15:35:26 -05:00
|
|
|
|
|
2015-05-18 20:56:00 +02:00
|
|
|
|
```rust
|
2015-02-19 15:35:26 -05:00
|
|
|
|
/// this
|
|
|
|
|
# fn foo() {}
|
|
|
|
|
|
|
|
|
|
#[doc="this"]
|
|
|
|
|
# fn bar() {}
|
|
|
|
|
```
|
2015-01-21 14:59:25 -05:00
|
|
|
|
|
2015-02-19 15:35:26 -05:00
|
|
|
|
are the same, as are these:
|
2015-01-21 14:59:25 -05:00
|
|
|
|
|
2015-05-18 20:56:00 +02:00
|
|
|
|
```rust
|
2015-02-19 15:35:26 -05:00
|
|
|
|
//! this
|
2015-01-21 14:59:25 -05:00
|
|
|
|
|
2015-10-24 20:00:58 +01:00
|
|
|
|
#![doc="this"]
|
2015-02-19 15:35:26 -05:00
|
|
|
|
```
|
2015-01-21 14:59:25 -05:00
|
|
|
|
|
2015-02-19 15:35:26 -05:00
|
|
|
|
You won't often see this attribute used for writing documentation, but it
|
|
|
|
|
can be useful when changing some options, or when writing a macro.
|
2015-02-09 22:24:49 -05:00
|
|
|
|
|
2015-02-19 15:35:26 -05:00
|
|
|
|
### Re-exports
|
2015-02-09 22:24:49 -05:00
|
|
|
|
|
2015-03-29 18:22:01 +08:00
|
|
|
|
`rustdoc` will show the documentation for a public re-export in both places:
|
2015-02-09 22:24:49 -05:00
|
|
|
|
|
2015-02-19 15:35:26 -05:00
|
|
|
|
```ignore
|
2015-02-09 22:24:49 -05:00
|
|
|
|
extern crate foo;
|
|
|
|
|
|
|
|
|
|
pub use foo::bar;
|
|
|
|
|
```
|
|
|
|
|
|
2015-10-13 15:59:59 +09:00
|
|
|
|
This will create documentation for `bar` both inside the documentation for the
|
2015-02-19 15:35:26 -05:00
|
|
|
|
crate `foo`, as well as the documentation for your crate. It will use the same
|
|
|
|
|
documentation in both places.
|
2015-02-09 22:24:49 -05:00
|
|
|
|
|
2015-03-28 18:09:51 +03:00
|
|
|
|
This behavior can be suppressed with `no_inline`:
|
2015-02-09 22:24:49 -05:00
|
|
|
|
|
2015-02-19 15:35:26 -05:00
|
|
|
|
```ignore
|
2015-02-09 22:24:49 -05:00
|
|
|
|
extern crate foo;
|
|
|
|
|
|
|
|
|
|
#[doc(no_inline)]
|
|
|
|
|
pub use foo::bar;
|
|
|
|
|
```
|
2015-02-19 15:35:26 -05:00
|
|
|
|
|
2015-10-09 03:00:43 +02:00
|
|
|
|
## Missing documentation
|
|
|
|
|
|
2015-10-09 14:04:45 +02:00
|
|
|
|
Sometimes you want to make sure that every single public thing in your project
|
|
|
|
|
is documented, especially when you are working on a library. Rust allows you to
|
2015-10-09 03:00:43 +02:00
|
|
|
|
to generate warnings or errors, when an item is missing documentation.
|
|
|
|
|
To generate warnings you use `warn`:
|
|
|
|
|
|
|
|
|
|
```rust
|
|
|
|
|
#![warn(missing_docs)]
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
And to generate errors you use `deny`:
|
|
|
|
|
|
|
|
|
|
```rust,ignore
|
|
|
|
|
#![deny(missing_docs)]
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
There are cases where you want to disable these warnings/errors to explicitly
|
|
|
|
|
leave something undocumented. This is done by using `allow`:
|
|
|
|
|
|
|
|
|
|
```rust
|
|
|
|
|
#[allow(missing_docs)]
|
|
|
|
|
struct Undocumented;
|
|
|
|
|
```
|
|
|
|
|
|
2015-10-09 14:08:32 +02:00
|
|
|
|
You might even want to hide items from the documentation completely:
|
|
|
|
|
|
|
|
|
|
```rust
|
|
|
|
|
#[doc(hidden)]
|
|
|
|
|
struct Hidden;
|
|
|
|
|
```
|
|
|
|
|
|
2015-02-19 15:35:26 -05:00
|
|
|
|
### Controlling HTML
|
|
|
|
|
|
|
|
|
|
You can control a few aspects of the HTML that `rustdoc` generates through the
|
|
|
|
|
`#![doc]` version of the attribute:
|
|
|
|
|
|
2015-05-18 20:56:00 +02:00
|
|
|
|
```rust
|
2015-08-09 14:15:05 -07:00
|
|
|
|
#![doc(html_logo_url = "https://www.rust-lang.org/logos/rust-logo-128x128-blk-v2.png",
|
|
|
|
|
html_favicon_url = "https://www.rust-lang.org/favicon.ico",
|
|
|
|
|
html_root_url = "https://doc.rust-lang.org/")]
|
2015-02-19 15:35:26 -05:00
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
This sets a few different options, with a logo, favicon, and a root URL.
|
|
|
|
|
|
|
|
|
|
## Generation options
|
|
|
|
|
|
2015-04-25 16:46:34 +02:00
|
|
|
|
`rustdoc` also contains a few other options on the command line, for further customization:
|
2015-02-19 15:35:26 -05:00
|
|
|
|
|
|
|
|
|
- `--html-in-header FILE`: includes the contents of FILE at the end of the
|
|
|
|
|
`<head>...</head>` section.
|
|
|
|
|
- `--html-before-content FILE`: includes the contents of FILE directly after
|
|
|
|
|
`<body>`, before the rendered content (including the search bar).
|
|
|
|
|
- `--html-after-content FILE`: includes the contents of FILE after all the rendered content.
|
2015-04-16 15:43:56 -04:00
|
|
|
|
|
|
|
|
|
## Security note
|
|
|
|
|
|
|
|
|
|
The Markdown in documentation comments is placed without processing into
|
|
|
|
|
the final webpage. Be careful with literal HTML:
|
|
|
|
|
|
|
|
|
|
```rust
|
|
|
|
|
/// <script>alert(document.cookie)</script>
|
|
|
|
|
# fn foo() {}
|
|
|
|
|
```
|