Skip to content
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

Greatly speed up doctests by compiling compatible doctests in one file #126245

Merged
merged 48 commits into from
Aug 14, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
48 commits
Select commit Hold shift + click to select a range
05fbfde
Clean up rustdoc make_test function code
GuillaumeGomez Jun 8, 2024
5e24437
Add `DocTest` type
GuillaumeGomez Jun 8, 2024
7d72482
Simplify doctest tests
GuillaumeGomez Jun 8, 2024
39f029a
Split doctests between standalone and mergeable ones
GuillaumeGomez Jun 8, 2024
96051f2
Split standalone and mergeable doctests
GuillaumeGomez Jun 8, 2024
6ae3524
Split doctests into two categories: mergeable ones and standalone ones
GuillaumeGomez Jun 10, 2024
58cd70e
Only merge doctests starting 2024 edition
GuillaumeGomez Jun 10, 2024
3147520
Add new doc codeblock `standalone` attribute
GuillaumeGomez Jun 10, 2024
23badff
Add documentation for the doctest `standalone` attribute
GuillaumeGomez Apr 30, 2024
b7079c5
Prevent merged doctests to break stdin if the generated file is too big
GuillaumeGomez Jun 10, 2024
a0ae8ac
If there is any AST error with a doctest, we make it a standalone test
GuillaumeGomez Jun 10, 2024
59a9e09
Correctly handle the case where there is no doctests to run
GuillaumeGomez Jun 10, 2024
7ec3cab
Correctly handle doctests with invalid AST
GuillaumeGomez Jun 10, 2024
2fd8d1c
Add/update `rustdoc-ui` tests to check new merged doctests
GuillaumeGomez Jun 10, 2024
010731d
Add new `run-make` tests for doctests
GuillaumeGomez Jun 10, 2024
03118fa
Simplify `has_main_fn` to be a boolean instead of a `Option<Span>`
GuillaumeGomez Jun 10, 2024
6eabffb
Greatly improve handling of doctests attributes, making it possible t…
GuillaumeGomez Jun 12, 2024
b6831bb
Don't merge doctests with `#[global_allocator]`
GuillaumeGomez Jun 13, 2024
dcc77b4
Fix weird memory allocation failure in merged doctests by storing doc…
GuillaumeGomez Jun 13, 2024
0f0681e
Make merged doctests run in their own process
GuillaumeGomez Jun 13, 2024
995858e
Don't change indent in merged doctests
GuillaumeGomez Jun 14, 2024
475824d
Mark location doctest as standalone since file information will not w…
GuillaumeGomez Jun 14, 2024
ec4d910
Correctly handle `internal_features` attribute
GuillaumeGomez Jun 15, 2024
c5ae545
If no argument is provided to merged doctests binary, they will be ru…
GuillaumeGomez Jun 18, 2024
0bd2c99
Only show rustdoc doctest compilation output if `nocapture` is used
GuillaumeGomez Jun 18, 2024
112e447
Disable merged doctests by default
GuillaumeGomez Jun 18, 2024
14f80f9
Correctly handle macros using `$crate` in merged doctests
GuillaumeGomez Jun 19, 2024
84d9b67
Add 2024 edition doctests to cover corner cases
GuillaumeGomez Jun 19, 2024
6d552ba
Improve code by removing unneeded function arguments
GuillaumeGomez Jun 20, 2024
bfabf1d
Add documentation on `DocTest` and `RunnableDoctest` structs
GuillaumeGomez Jun 20, 2024
d512438
Run mergeable doctest as part of standalone doctests if there is only…
GuillaumeGomez Jun 20, 2024
1e1743a
Reduce merged doctest source code size
GuillaumeGomez Jun 20, 2024
ffc18c5
Rename `DocTest` into `DocTestBuilder`
GuillaumeGomez Jun 23, 2024
0fbc32c
If there are crate attributes, we prevent doctest to be merged with o…
GuillaumeGomez Jun 23, 2024
a3bc2c7
Improve code readability
GuillaumeGomez Jun 24, 2024
72aeffa
Remove need for `unsafe` code in merged doctests
GuillaumeGomez Jul 8, 2024
4b1db07
Don't special-case if there is only one merged doctest
GuillaumeGomez Jul 8, 2024
ab3d90e
Unify naming of `DocTest`
GuillaumeGomez Jul 9, 2024
baf8ce8
Move `is_multiple_tests` argument into `RunnableDocTest`
GuillaumeGomez Jul 9, 2024
c9f730e
Improve documentation for internal doctest API
GuillaumeGomez Jul 9, 2024
cbf6fe0
Add more merged doctests tests
GuillaumeGomez Jul 9, 2024
7dcb841
Add doctest to ensure that doctests with crate-level attributes are n…
GuillaumeGomez Jul 11, 2024
903d2db
Only keep "useful" code in `tests/rustdoc-ui/2024-doctests-checks.rs`
GuillaumeGomez Jul 11, 2024
a708d0b
Fix commands syntax in rustdoc-ui tests
GuillaumeGomez Jul 12, 2024
f1c1c49
Run fmt
GuillaumeGomez Jul 29, 2024
488614d
Update `tests/run-make/doctests-keep-binaries-2024/rmake.rs` test to …
GuillaumeGomez Jul 29, 2024
1d75f78
Ignore cross compile check for `tests/run-make/doctests-keep-binaries…
GuillaumeGomez Aug 7, 2024
05fb8ff
Fix intra-doc link
GuillaumeGomez Aug 12, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion library/core/src/panic/location.rs
Original file line number Diff line number Diff line change
Expand Up @@ -44,7 +44,7 @@ impl<'a> Location<'a> {
///
/// # Examples
///
/// ```
/// ```standalone
/// use std::panic::Location;
///
/// /// Returns the [`Location`] at which it is called.
Expand Down
34 changes: 34 additions & 0 deletions library/test/src/types.rs
Original file line number Diff line number Diff line change
Expand Up @@ -250,3 +250,37 @@ pub struct TestDescAndFn {
pub desc: TestDesc,
pub testfn: TestFn,
}

impl TestDescAndFn {
pub const fn new_doctest(
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Is libtest considered perma-unstable? Just want to make sure we're not affecting the public API. Either way, we should probably have some from T-libs sign-off on this change.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Per @Amanieu and @thomcc, libtest is neither stable nor intended to be stable.

I believe that should be sufficient given that these remarks reflect an existing consensus.

test_name: &'static str,
ignore: bool,
source_file: &'static str,
start_line: usize,
no_run: bool,
should_panic: bool,
testfn: TestFn,
) -> Self {
Self {
desc: TestDesc {
name: StaticTestName(test_name),
ignore,
ignore_message: None,
source_file,
start_line,
start_col: 0,
end_line: 0,
end_col: 0,
compile_fail: false,
no_run,
should_panic: if should_panic {
options::ShouldPanic::Yes
} else {
options::ShouldPanic::No
},
test_type: TestType::DocTest,
},
testfn,
}
}
}
51 changes: 51 additions & 0 deletions src/doc/rustdoc/src/write-documentation/documentation-tests.md
Original file line number Diff line number Diff line change
Expand Up @@ -376,6 +376,57 @@ that the code sample should be compiled using the respective edition of Rust.
# fn foo() {}
```

Starting in the 2024 edition[^edition-note], compatible doctests are merged as one before being
run. We combine doctests for performance reasons: the slowest part of doctests is to compile them.
Merging all of them into one file and compiling this new file, then running the doctests is much
faster. Whether doctests are merged or not, they are run in their own process.

An example of time spent when running doctests:

[sysinfo crate](https://crates.io/crates/sysinfo):

```text
wall-time duration: 4.59s
total compile time: 27.067s
total runtime: 3.969s
```

Rust core library:

```text
wall-time duration: 102s
total compile time: 775.204s
total runtime: 15.487s
```

[^edition-note]: This is based on the edition of the whole crate, not the edition of the individual
test case that may be specified in its code attribute.

In some cases, doctests cannot be merged. For example, if you have:

```rust
//! ```
//! let location = std::panic::Location::caller();
//! assert_eq!(location.line(), 4);
//! ```
```

The problem with this code is that, if you change any other doctests, it'll likely break when
runing `rustdoc --test`, making it tricky to maintain.

This is where the `standalone` attribute comes in: it tells `rustdoc` that a doctest
should not be merged with the others. So the previous code should use it:

```rust
//! ```standalone
//! let location = std::panic::Location::caller();
//! assert_eq!(location.line(), 4);
//! ```
```

In this case, it means that the line information will not change if you add/remove other
doctests.

### Custom CSS classes for code blocks

```rust
Expand Down
Loading
Loading