Skip to content

Commit 123ea25

Browse files
Correctly handle line comments in attributes and generate extern crates
outside of wrapping function
1 parent 6f7e8d4 commit 123ea25

File tree

7 files changed

+98
-25
lines changed

7 files changed

+98
-25
lines changed

src/librustdoc/doctest/make.rs

+35-15
Original file line numberDiff line numberDiff line change
@@ -176,9 +176,24 @@ impl DocTestBuilder {
176176

177177
// Now push any outer attributes from the example, assuming they
178178
// are intended to be crate attributes.
179-
prog.push_str(&self.crate_attrs);
180-
prog.push_str(&self.maybe_crate_attrs);
181-
prog.push_str(&self.crates);
179+
if !self.crate_attrs.is_empty() {
180+
prog.push_str(&self.crate_attrs);
181+
if !self.crate_attrs.ends_with('\n') {
182+
prog.push('\n');
183+
}
184+
}
185+
if !self.maybe_crate_attrs.is_empty() {
186+
prog.push_str(&self.maybe_crate_attrs);
187+
if !self.maybe_crate_attrs.ends_with('\n') {
188+
prog.push('\n');
189+
}
190+
}
191+
if !self.crates.is_empty() {
192+
prog.push_str(&self.crates);
193+
if !self.crates.ends_with('\n') {
194+
prog.push('\n');
195+
}
196+
}
182197

183198
// Don't inject `extern crate std` because it's already injected by the
184199
// compiler.
@@ -276,7 +291,6 @@ const DOCTEST_CODE_WRAPPER: &str = "fn f(){";
276291
fn parse_source(source: &str, crate_name: &Option<&str>) -> Result<ParseSourceInfo, ()> {
277292
use rustc_errors::DiagCtxt;
278293
use rustc_errors::emitter::{Emitter, HumanEmitter};
279-
// use rustc_parse::parser::ForceCollect;
280294
use rustc_span::source_map::FilePathMapping;
281295

282296
let mut info =
@@ -338,7 +352,8 @@ fn parse_source(source: &str, crate_name: &Option<&str>) -> Result<ParseSourceIn
338352
info: &mut ParseSourceInfo,
339353
crate_name: &Option<&str>,
340354
is_top_level: bool,
341-
) {
355+
) -> bool {
356+
let mut is_crate = false;
342357
if !info.has_global_allocator
343358
&& item.attrs.iter().any(|attr| attr.name_or_empty() == sym::global_allocator)
344359
{
@@ -354,12 +369,13 @@ fn parse_source(source: &str, crate_name: &Option<&str>) -> Result<ParseSourceIn
354369
if let Some(ref body) = fn_item.body {
355370
for stmt in &body.stmts {
356371
if let StmtKind::Item(ref item) = stmt.kind {
357-
check_item(item, info, crate_name, false)
372+
is_crate |= check_item(item, info, crate_name, false);
358373
}
359374
}
360375
}
361376
}
362377
ast::ItemKind::ExternCrate(original) => {
378+
is_crate = true;
363379
if !info.found_extern_crate
364380
&& let Some(crate_name) = crate_name
365381
{
@@ -374,6 +390,7 @@ fn parse_source(source: &str, crate_name: &Option<&str>) -> Result<ParseSourceIn
374390
}
375391
_ => {}
376392
}
393+
is_crate
377394
}
378395

379396
let mut prev_span_hi = None;
@@ -412,8 +429,11 @@ fn parse_source(source: &str, crate_name: &Option<&str>) -> Result<ParseSourceIn
412429
}
413430
}
414431
for stmt in &body.stmts {
432+
let mut is_crate = false;
415433
match stmt.kind {
416-
StmtKind::Item(ref item) => check_item(&item, &mut info, crate_name, true),
434+
StmtKind::Item(ref item) => {
435+
is_crate = check_item(&item, &mut info, crate_name, true);
436+
}
417437
StmtKind::Expr(ref expr) if matches!(expr.kind, ast::ExprKind::Err(_)) => {
418438
cancel_error_count(&psess);
419439
return Err(());
@@ -450,15 +470,15 @@ fn parse_source(source: &str, crate_name: &Option<&str>) -> Result<ParseSourceIn
450470
if info.everything_else.is_empty()
451471
&& (!info.maybe_crate_attrs.is_empty() || !info.crate_attrs.is_empty())
452472
{
453-
// We add potential backlines/comments into attributes if there are some.
454-
push_to_s(
455-
&mut info.maybe_crate_attrs,
456-
source,
457-
span.shrink_to_lo(),
458-
&mut prev_span_hi,
459-
);
473+
// We add potential backlines/comments if there are some in items generated
474+
// before the wrapping function.
475+
push_to_s(&mut info.crates, source, span.shrink_to_lo(), &mut prev_span_hi);
476+
}
477+
if !is_crate {
478+
push_to_s(&mut info.everything_else, source, span, &mut prev_span_hi);
479+
} else {
480+
push_to_s(&mut info.crates, source, span, &mut prev_span_hi);
460481
}
461-
push_to_s(&mut info.everything_else, source, span, &mut prev_span_hi);
462482
}
463483
Ok(info)
464484
}

src/librustdoc/doctest/tests.rs

+27-3
Original file line numberDiff line numberDiff line change
@@ -127,8 +127,8 @@ fn make_test_manual_extern_crate() {
127127
use asdf::qwop;
128128
assert_eq!(2+2, 4);";
129129
let expected = "#![allow(unused)]
130-
fn main() {
131130
extern crate asdf;
131+
fn main() {
132132
use asdf::qwop;
133133
assert_eq!(2+2, 4);
134134
}"
@@ -144,8 +144,8 @@ fn make_test_manual_extern_crate_with_macro_use() {
144144
use asdf::qwop;
145145
assert_eq!(2+2, 4);";
146146
let expected = "#![allow(unused)]
147-
fn main() {
148147
#[macro_use] extern crate asdf;
148+
fn main() {
149149
use asdf::qwop;
150150
assert_eq!(2+2, 4);
151151
}"
@@ -197,6 +197,7 @@ fn make_test_crate_attrs() {
197197
assert_eq!(2+2, 4);";
198198
let expected = "#![allow(unused)]
199199
#![feature(sick_rad)]
200+
200201
fn main() {
201202
assert_eq!(2+2, 4);
202203
}"
@@ -277,10 +278,10 @@ fn make_test_issues_33731() {
277278
assert_eq!(asdf::foo, 4);";
278279

279280
let expected = "#![allow(unused)]
281+
extern crate hella_qwop;
280282
#[allow(unused_extern_crates)]
281283
extern crate r#asdf;
282284
fn main() {
283-
extern crate hella_qwop;
284285
assert_eq!(asdf::foo, 4);
285286
}"
286287
.to_string();
@@ -401,3 +402,26 @@ fn check_split_args() {
401402
compare("a\n\t \rb", &["a", "b"]);
402403
compare("a\n\t1 \rb", &["a", "1", "b"]);
403404
}
405+
406+
#[test]
407+
fn comment_in_attrs() {
408+
// if input already has a fn main, it should insert a space before it
409+
let opts = default_global_opts("");
410+
let input = "\
411+
#![feature(rustdoc_internals)]
412+
#![allow(internal_features)]
413+
#![doc(rust_logo)]
414+
//! This crate has the Rust(tm) branding on it.";
415+
let expected = "\
416+
#![allow(unused)]
417+
#![feature(rustdoc_internals)]
418+
#![allow(internal_features)]
419+
#![doc(rust_logo)]
420+
//! This crate has the Rust(tm) branding on it.
421+
fn main() {
422+
423+
}"
424+
.to_string();
425+
let (output, len) = make_test(input, None, false, &opts, None);
426+
assert_eq!((output, len), (expected, 2));
427+
}

tests/run-make/rustdoc-error-lines/rmake.rs

+3-3
Original file line numberDiff line numberDiff line change
@@ -8,11 +8,11 @@ fn main() {
88

99
let should_contain = &[
1010
"input.rs - foo (line 5)",
11-
"input.rs:7:15",
11+
"input.rs:8:15",
1212
"input.rs - bar (line 13)",
13-
"input.rs:15:15",
13+
"input.rs:16:15",
1414
"input.rs - bar (line 22)",
15-
"input.rs:24:15",
15+
"input.rs:25:15",
1616
];
1717
for text in should_contain {
1818
assert!(output.contains(text), "output doesn't contains {:?}", text);

tests/rustdoc-ui/doctest/display-output.stdout

+3-3
Original file line numberDiff line numberDiff line change
@@ -6,7 +6,7 @@ successes:
66

77
---- $DIR/display-output.rs - foo (line 9) stdout ----
88
warning: unused variable: `x`
9-
--> $DIR/display-output.rs:11:5
9+
--> $DIR/display-output.rs:12:5
1010
|
1111
LL | let x = 12;
1212
| ^ help: if this is intentional, prefix it with an underscore: `_x`
@@ -19,13 +19,13 @@ LL | #![warn(unused)]
1919
= note: `#[warn(unused_variables)]` implied by `#[warn(unused)]`
2020

2121
warning: unused variable: `x`
22-
--> $DIR/display-output.rs:13:8
22+
--> $DIR/display-output.rs:14:8
2323
|
2424
LL | fn foo(x: &dyn std::fmt::Display) {}
2525
| ^ help: if this is intentional, prefix it with an underscore: `_x`
2626

2727
warning: function `foo` is never used
28-
--> $DIR/display-output.rs:13:4
28+
--> $DIR/display-output.rs:14:4
2929
|
3030
LL | fn foo(x: &dyn std::fmt::Display) {}
3131
| ^^^
+23
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,23 @@
1+
//@ check-pass
2+
//@ compile-flags:--test --test-args=--test-threads=1
3+
//@ normalize-stdout: "tests/rustdoc-ui/doctest" -> "$$DIR"
4+
//@ normalize-stdout: "finished in \d+\.\d+s" -> "finished in $$TIME"
5+
6+
// This test ensures that crate imports are placed outside of the `main` function
7+
// so they work all the time (even in 2015 edition).
8+
9+
/// ```rust
10+
/// #![feature(test)]
11+
///
12+
/// extern crate test;
13+
/// use test::Bencher;
14+
///
15+
/// #[bench]
16+
/// fn bench_xor_1000_ints(b: &mut Bencher) {
17+
/// b.iter(|| {
18+
/// (0..1000).fold(0, |old, new| old ^ new);
19+
/// });
20+
/// }
21+
/// ```
22+
///
23+
pub fn foo() {}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,6 @@
1+
2+
running 1 test
3+
test $DIR/extern-crate.rs - foo (line 9) ... ok
4+
5+
test result: ok. 1 passed; 0 failed; 0 ignored; 0 measured; 0 filtered out; finished in $TIME
6+

tests/rustdoc/playground.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -24,4 +24,4 @@
2424
2525
//@ matches foo/index.html '//a[@class="test-arrow"][@href="https://www.example.com/?code=%23!%5Ballow(unused)%5D%0Afn+main()+%7B%0A++++println!(%22Hello,+world!%22);%0A%7D&edition=2015"]' ""
2626
//@ matches foo/index.html '//a[@class="test-arrow"][@href="https://www.example.com/?code=%23!%5Ballow(unused)%5D%0Afn+main()+%7B%0A++++println!(%22Hello,+world!%22);%0A%7D&edition=2015"]' ""
27-
//@ matches foo/index.html '//a[@class="test-arrow"][@href="https://www.example.com/?code=%23!%5Ballow(unused)%5D%0A%23!%5Bfeature(something)%5D%0A%0Afn+main()+%7B%0A++++println!(%22Hello,+world!%22);%0A%7D&version=nightly&edition=2015"]' ""
27+
//@ matches foo/index.html '//a[@class="test-arrow"][@href="https://www.example.com/?code=%23!%5Ballow(unused)%5D%0A%23!%5Bfeature(something)%5D%0A%0A%0Afn+main()+%7B%0A++++println!(%22Hello,+world!%22);%0A%7D&version=nightly&edition=2015"]' ""

0 commit comments

Comments
 (0)