Rollup merge of #56793 - QuietMisdreavus:better-doctests, r=GuillaumeGomez

rustdoc: look for comments when scraping attributes/crates from doctests

Fixes https://github.com/rust-lang/rust/issues/56727

When scraping out crate-level attributes and `extern crate` statements, we wouldn't look for comments, so any presence of comments would shunt it and everything after it into "everything else". This could cause parsing issues when looking for `fn main` and `extern crate my_crate` later on, which would in turn cause rustdoc to incorrectly wrap a test with `fn main` when it already had one declared.

I took the opportunity to clean up the logic a little bit, but it would still benefit from a libsyntax-based loop like the `fn main` detection.
This commit is contained in:
Mazdak Farrokhzad 2018-12-16 14:08:30 +01:00 committed by GitHub
commit 32a6a95f41
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
2 changed files with 88 additions and 16 deletions

View File

@ -395,6 +395,7 @@ pub fn make_test(s: &str,
// Now push any outer attributes from the example, assuming they // Now push any outer attributes from the example, assuming they
// are intended to be crate attributes. // are intended to be crate attributes.
prog.push_str(&crate_attrs); prog.push_str(&crate_attrs);
prog.push_str(&crates);
// Uses libsyntax to parse the doctest and find if there's a main fn and the extern // Uses libsyntax to parse the doctest and find if there's a main fn and the extern
// crate already is included. // crate already is included.
@ -488,37 +489,78 @@ pub fn make_test(s: &str,
prog.push_str("\n}"); prog.push_str("\n}");
} }
debug!("final doctest:\n{}", prog);
(prog, line_offset) (prog, line_offset)
} }
// FIXME(aburka): use a real parser to deal with multiline attributes // FIXME(aburka): use a real parser to deal with multiline attributes
fn partition_source(s: &str) -> (String, String, String) { fn partition_source(s: &str) -> (String, String, String) {
let mut after_header = false; #[derive(Copy, Clone, PartialEq)]
enum PartitionState {
Attrs,
Crates,
Other,
}
let mut state = PartitionState::Attrs;
let mut before = String::new(); let mut before = String::new();
let mut crates = String::new(); let mut crates = String::new();
let mut after = String::new(); let mut after = String::new();
for line in s.lines() { for line in s.lines() {
let trimline = line.trim(); let trimline = line.trim();
let header = trimline.chars().all(|c| c.is_whitespace()) ||
trimline.starts_with("#![") || // FIXME(misdreavus): if a doc comment is placed on an extern crate statement, it will be
trimline.starts_with("#[macro_use] extern crate") || // shunted into "everything else"
trimline.starts_with("extern crate"); match state {
if !header || after_header { PartitionState::Attrs => {
after_header = true; state = if trimline.starts_with("#![") ||
after.push_str(line); trimline.chars().all(|c| c.is_whitespace()) ||
after.push_str("\n"); (trimline.starts_with("//") && !trimline.starts_with("///"))
} else { {
if trimline.starts_with("#[macro_use] extern crate") PartitionState::Attrs
|| trimline.starts_with("extern crate") { } else if trimline.starts_with("extern crate") ||
trimline.starts_with("#[macro_use] extern crate")
{
PartitionState::Crates
} else {
PartitionState::Other
};
}
PartitionState::Crates => {
state = if trimline.starts_with("extern crate") ||
trimline.starts_with("#[macro_use] extern crate") ||
trimline.chars().all(|c| c.is_whitespace()) ||
(trimline.starts_with("//") && !trimline.starts_with("///"))
{
PartitionState::Crates
} else {
PartitionState::Other
};
}
PartitionState::Other => {}
}
match state {
PartitionState::Attrs => {
before.push_str(line);
before.push_str("\n");
}
PartitionState::Crates => {
crates.push_str(line); crates.push_str(line);
crates.push_str("\n"); crates.push_str("\n");
} }
before.push_str(line); PartitionState::Other => {
before.push_str("\n"); after.push_str(line);
after.push_str("\n");
}
} }
} }
debug!("before:\n{}", before);
debug!("crates:\n{}", crates);
debug!("after:\n{}", after);
(before, after, crates) (before, after, crates)
} }
@ -1035,8 +1077,8 @@ fn main() {
assert_eq!(2+2, 4);"; assert_eq!(2+2, 4);";
let expected = let expected =
"#![allow(unused)] "#![allow(unused)]
fn main() {
//Ceci n'est pas une `fn main` //Ceci n'est pas une `fn main`
fn main() {
assert_eq!(2+2, 4); assert_eq!(2+2, 4);
}".to_string(); }".to_string();
let output = make_test(input, None, false, &opts); let output = make_test(input, None, false, &opts);
@ -1083,8 +1125,8 @@ assert_eq!(2+2, 4);";
let expected = let expected =
"#![allow(unused)] "#![allow(unused)]
fn main() {
// fn main // fn main
fn main() {
assert_eq!(2+2, 4); assert_eq!(2+2, 4);
}".to_string(); }".to_string();

View File

@ -0,0 +1,30 @@
// Copyright 2018 The Rust Project Developers. See the COPYRIGHT
// file at the top-level directory of this distribution and at
// http://rust-lang.org/COPYRIGHT.
//
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
// option. This file may not be copied, modified, or distributed
// except according to those terms.
// compile-flags:--test
// comments, both doc comments and regular ones, used to trick rustdoc's doctest parser into
// thinking that everything after it was part of the regular program. combined with the libsyntax
// parser loop failing to detect the manual main function, it would wrap everything in `fn main`,
// which would cause the doctest to fail as the "extern crate" declaration was no longer valid.
// oddly enough, it would pass in 2018 if a crate was in the extern prelude. see
// https://github.com/rust-lang/rust/issues/56727
//! ```
//! // crate: proc-macro-test
//! //! this is a test
//!
//! // used to pull in proc-macro specific items
//! extern crate proc_macro;
//!
//! use proc_macro::TokenStream;
//!
//! # fn main() {}
//! ```