mirror of
https://github.com/rust-lang/rust.git
synced 2024-11-02 15:32:06 +00:00
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:
commit
32a6a95f41
@ -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();
|
||||||
|
|
||||||
|
30
src/test/rustdoc/comment-in-doctest.rs
Normal file
30
src/test/rustdoc/comment-in-doctest.rs
Normal 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() {}
|
||||||
|
//! ```
|
Loading…
Reference in New Issue
Block a user