Skip to content

Commit 9451a61

Browse files
authored
Rollup merge of #134260 - GuillaumeGomez:doctest-attrs, r=notriddle
Correctly handle comments in attributes in doctests source code Fixes #134221. The problem was that attributes are "inlined" (backlines are stripped), then when there is an inline comment inside it, the attribute is never considered valid (since unclosed). Fix was to simply put back backlines in case it's a multiline attribute. r? ``@notriddle``
2 parents d03e3c8 + c367cc3 commit 9451a61

File tree

5 files changed

+165
-18
lines changed

5 files changed

+165
-18
lines changed

src/librustdoc/doctest/make.rs

+42-18
Original file line numberDiff line numberDiff line change
@@ -51,8 +51,17 @@ impl DocTestBuilder {
5151
!lang_str.compile_fail && !lang_str.test_harness && !lang_str.standalone_crate
5252
});
5353

54-
let SourceInfo { crate_attrs, maybe_crate_attrs, crates, everything_else } =
55-
partition_source(source, edition);
54+
let Some(SourceInfo { crate_attrs, maybe_crate_attrs, crates, everything_else }) =
55+
partition_source(source, edition)
56+
else {
57+
return Self::invalid(
58+
String::new(),
59+
String::new(),
60+
String::new(),
61+
source.to_string(),
62+
test_id,
63+
);
64+
};
5665

5766
// Uses librustc_ast to parse the doctest and find if there's a main fn and the extern
5867
// crate already is included.
@@ -77,18 +86,7 @@ impl DocTestBuilder {
7786
else {
7887
// If the parser panicked due to a fatal error, pass the test code through unchanged.
7988
// The error will be reported during compilation.
80-
return Self {
81-
supports_color: false,
82-
has_main_fn: false,
83-
crate_attrs,
84-
maybe_crate_attrs,
85-
crates,
86-
everything_else,
87-
already_has_extern_crate: false,
88-
test_id,
89-
failed_ast: true,
90-
can_be_merged: false,
91-
};
89+
return Self::invalid(crate_attrs, maybe_crate_attrs, crates, everything_else, test_id);
9290
};
9391
// If the AST returned an error, we don't want this doctest to be merged with the
9492
// others. Same if it contains `#[feature]` or `#[no_std]`.
@@ -113,6 +111,27 @@ impl DocTestBuilder {
113111
}
114112
}
115113

114+
fn invalid(
115+
crate_attrs: String,
116+
maybe_crate_attrs: String,
117+
crates: String,
118+
everything_else: String,
119+
test_id: Option<String>,
120+
) -> Self {
121+
Self {
122+
supports_color: false,
123+
has_main_fn: false,
124+
crate_attrs,
125+
maybe_crate_attrs,
126+
crates,
127+
everything_else,
128+
already_has_extern_crate: false,
129+
test_id,
130+
failed_ast: true,
131+
can_be_merged: false,
132+
}
133+
}
134+
116135
/// Transforms a test into code that can be compiled into a Rust binary, and returns the number of
117136
/// lines before the test code begins.
118137
pub(crate) fn generate_unique_doctest(
@@ -518,8 +537,8 @@ fn handle_attr(mod_attr_pending: &mut String, source_info: &mut SourceInfo, edit
518537
push_to.push('\n');
519538
// If it's complete, then we can clear the pending content.
520539
mod_attr_pending.clear();
521-
} else if mod_attr_pending.ends_with('\\') {
522-
mod_attr_pending.push('n');
540+
} else {
541+
mod_attr_pending.push_str("\n");
523542
}
524543
}
525544

@@ -531,7 +550,7 @@ struct SourceInfo {
531550
everything_else: String,
532551
}
533552

534-
fn partition_source(s: &str, edition: Edition) -> SourceInfo {
553+
fn partition_source(s: &str, edition: Edition) -> Option<SourceInfo> {
535554
#[derive(Copy, Clone, PartialEq)]
536555
enum PartitionState {
537556
Attrs,
@@ -606,11 +625,16 @@ fn partition_source(s: &str, edition: Edition) -> SourceInfo {
606625
}
607626
}
608627

628+
if !mod_attr_pending.is_empty() {
629+
debug!("invalid doctest code: {s:?}");
630+
return None;
631+
}
632+
609633
source_info.everything_else = source_info.everything_else.trim().to_string();
610634

611635
debug!("crate_attrs:\n{}{}", source_info.crate_attrs, source_info.maybe_crate_attrs);
612636
debug!("crates:\n{}", source_info.crates);
613637
debug!("after:\n{}", source_info.everything_else);
614638

615-
source_info
639+
Some(source_info)
616640
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,15 @@
1+
//@ compile-flags:--test --test-args --test-threads=1
2+
//@ failure-status: 101
3+
//@ normalize-stdout-test: "tests/rustdoc-ui/doctest" -> "$$DIR"
4+
//@ normalize-stdout-test: "finished in \d+\.\d+s" -> "finished in $$TIME"
5+
//@ normalize-stdout-test: ".rs:\d+:\d+" -> ".rs:$$LINE:$$COL"
6+
7+
//! ```
8+
#![doc = "#![all\
9+
ow(unused)]"]
10+
//! ```
11+
//!
12+
//! ```
13+
#![doc = r#"#![all\
14+
ow(unused)]"#]
15+
//! ```
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,31 @@
1+
2+
running 2 tests
3+
test $DIR/comment-in-attr-134221-2.rs - (line 11) ... FAILED
4+
test $DIR/comment-in-attr-134221-2.rs - (line 7) ... ok
5+
6+
failures:
7+
8+
---- $DIR/comment-in-attr-134221-2.rs - (line 11) stdout ----
9+
error: unknown start of token: \
10+
--> $DIR/comment-in-attr-134221-2.rs:$LINE:$COL
11+
|
12+
LL | #![all\
13+
| ^
14+
15+
error: expected one of `(`, `::`, `=`, `[`, `]`, or `{`, found `ow`
16+
--> $DIR/comment-in-attr-134221-2.rs:$LINE:$COL
17+
|
18+
LL | #![all\
19+
| - expected one of `(`, `::`, `=`, `[`, `]`, or `{`
20+
LL | ow(unused)]
21+
| ^^ unexpected token
22+
23+
error: aborting due to 2 previous errors
24+
25+
Couldn't compile the test.
26+
27+
failures:
28+
$DIR/comment-in-attr-134221-2.rs - (line 11)
29+
30+
test result: FAILED. 1 passed; 1 failed; 0 ignored; 0 measured; 0 filtered out; finished in $TIME
31+
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,27 @@
1+
// Regression test for <https://github.com/rust-lang/rust/issues/134221>.
2+
// It checks that even if there are comments in the attributes, the attributes
3+
// will still be generated correctly (and therefore fail in this test).
4+
5+
//@ compile-flags:--test --test-args --test-threads=1
6+
//@ failure-status: 101
7+
//@ normalize-stdout-test: "tests/rustdoc-ui/doctest" -> "$$DIR"
8+
//@ normalize-stdout-test: "finished in \d+\.\d+s" -> "finished in $$TIME"
9+
//@ normalize-stdout-test: ".rs:\d+:\d+" -> ".rs:$$LINE:$$COL"
10+
11+
/*!
12+
```rust
13+
#![feature(
14+
foo, //
15+
)]
16+
```
17+
18+
```rust
19+
#![feature(
20+
foo,
21+
)]
22+
```
23+
24+
```rust
25+
#![
26+
```
27+
*/
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,50 @@
1+
2+
running 3 tests
3+
test $DIR/comment-in-attr-134221.rs - (line 11) ... FAILED
4+
test $DIR/comment-in-attr-134221.rs - (line 17) ... FAILED
5+
test $DIR/comment-in-attr-134221.rs - (line 23) ... FAILED
6+
7+
failures:
8+
9+
---- $DIR/comment-in-attr-134221.rs - (line 11) stdout ----
10+
error[E0635]: unknown feature `foo`
11+
--> $DIR/comment-in-attr-134221.rs:$LINE:$COL
12+
|
13+
LL | foo, //
14+
| ^^^
15+
16+
error: aborting due to 1 previous error
17+
18+
For more information about this error, try `rustc --explain E0635`.
19+
Couldn't compile the test.
20+
---- $DIR/comment-in-attr-134221.rs - (line 17) stdout ----
21+
error[E0635]: unknown feature `foo`
22+
--> $DIR/comment-in-attr-134221.rs:$LINE:$COL
23+
|
24+
LL | foo,
25+
| ^^^
26+
27+
error: aborting due to 1 previous error
28+
29+
For more information about this error, try `rustc --explain E0635`.
30+
Couldn't compile the test.
31+
---- $DIR/comment-in-attr-134221.rs - (line 23) stdout ----
32+
error: this file contains an unclosed delimiter
33+
--> $DIR/comment-in-attr-134221.rs:$LINE:$COL
34+
|
35+
LL | #![
36+
| -^
37+
| |
38+
| unclosed delimiter
39+
40+
error: aborting due to 1 previous error
41+
42+
Couldn't compile the test.
43+
44+
failures:
45+
$DIR/comment-in-attr-134221.rs - (line 11)
46+
$DIR/comment-in-attr-134221.rs - (line 17)
47+
$DIR/comment-in-attr-134221.rs - (line 23)
48+
49+
test result: FAILED. 0 passed; 3 failed; 0 ignored; 0 measured; 0 filtered out; finished in $TIME
50+

0 commit comments

Comments
 (0)